This repository has been archived by the owner on May 23, 2023. It is now read-only.
-
Notifications
You must be signed in to change notification settings - Fork 523
/
Copy pathreverse.lua
472 lines (411 loc) · 19.5 KB
/
reverse.lua
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
local abs, max, rad, sin = math.abs, math.max, math.rad, math.sin;
function courseplay:goReverse(vehicle,lx,lz,mode2)
-- TODO get rid of cx and cz and the global Waypoints array. I know this is horrible but I don't want to
-- maintain a cx/cz in the Course object
local getX = function(waypoints, ix)
return waypoints[ix].cx or waypoints[ix].x
end
local getZ = function(waypoints, ix)
return waypoints[ix].cz or waypoints[ix].z
end
local waypoints, index
-- when the AI Driver is driving we want to use the course set up by the driver and not the legacy
-- global variable
-- TODO: fix missing encapsulation
index = math.min(vehicle.cp.driver.ppc:getCurrentWaypointIx() + 1, vehicle.cp.driver.ppc.course:getNumberOfWaypoints())
waypoints = vehicle.cp.driver:getCurrentCourse().waypoints
local fwd = false;
local workTool = courseplay:getFirstReversingWheeledWorkTool(vehicle) or vehicle.cp.workTools[1];
local newTarget;
local attacherVehicle;
if vehicle.cp.turnTargets and vehicle.cp.curTurnIndex then
newTarget = vehicle.cp.turnTargets[vehicle.cp.curTurnIndex];
end;
if workTool then
attacherVehicle = workTool.getAttacherVehicle and workTool:getAttacherVehicle() or vehicle;
-- Attacher modules and HookLift modules that needs the hookLiftTrailer
if courseplay:isHookLift(workTool) or courseplay:isAttacherModule(workTool) then
workTool = attacherVehicle;
if workTool == vehicle and vehicle.cp.workTools[2] ~= nil then
workTool = vehicle.cp.workTools[2];
if courseplay:isAttacherModule(workTool) then
workTool = attacherVehicle;
end;
end;
end;
end;
local mode = vehicle.cp.settings.driverMode:get()
local debugActive = courseplay.debugChannels[courseplay.DBG_REVERSE];
local isNotValid = vehicle.cp.numWorkTools == 0 or workTool == nil or workTool.cp.isPivot == nil or not workTool.cp.frontNode or mode == courseplay.MODE_SHOVEL_FILL_AND_EMPTY;
if isNotValid then
-- Simple reversing, no trailer to back up, so set the direction and get out of here, no need for
-- all the sophisticated reversing
if newTarget then
-- If we have the revPosX, revPosZ set, use those
if newTarget.revPosX and newTarget.revPosZ then
local _, vehicleY, _ = getWorldTranslation(vehicle.cp.directionNode);
lx, lz = AIVehicleUtil.getDriveDirection(vehicle.cp.directionNode, newTarget.revPosX, vehicleY, newTarget.revPosZ);
end;
elseif mode ~= courseplay.MODE_SHOVEL_FILL_AND_EMPTY then
-- Start: Fixes issue #525
local tx, ty, tz = localToWorld(vehicle.cp.directionNode, 0, 1, -3);
local nx, ny, nz = localDirectionToWorld(vehicle.cp.directionNode, lx, -0,1, lz);
courseplay:doTriggerRaycasts(vehicle, 'tipTrigger', 'rev', false, tx, ty, tz, nx, ny, nz);
-- End: Fixes issue #525
end
-- false means that this is a trivial reverse and can be handled by drive
return -lx,-lz,fwd, false;
end;
local node = workTool.cp.realTurningNode;
if mode2 then
vehicle.cp.toolsRealTurningNode = node;
end
local xTipper,yTipper,zTipper = getWorldTranslation(node);
if debugActive then cpDebug:drawPoint(xTipper, yTipper+5, zTipper, 1, 0 , 0) end;
local frontNode = workTool.cp.frontNode;
local xFrontNode,yFrontNode,zFrontNode = getWorldTranslation(frontNode);
local tcx,tcy,tcz =0,0,0;
if debugActive and not newTarget then
cpDebug:drawPoint(xFrontNode,yFrontNode+3,zFrontNode, 1, 0 , 0);
if not vehicle.cp.checkReverseValdityPrinted then
local checkValidity = false;
for i=index, #waypoints do
if waypoints[i].rev then
tcx = getX(waypoints, i);
tcz = getZ(waypoints, i);
local _,_,z = worldToLocal(node, tcx,yTipper,tcz);
if z < 0 then
checkValidity = true;
break;
end;
else
break;
end;
end;
if not checkValidity then
print(nameNum(vehicle) ..": reverse course is not valid");
end;
vehicle.cp.checkReverseValdityPrinted = true;
end;
end;
if newTarget then
if newTarget.revPosX and newTarget.revPosZ then
tcx = newTarget.revPosX;
tcz = newTarget.revPosZ;
else
tcx = newTarget.posX;
tcz = newTarget.posZ;
end;
elseif not mode2 then
for i= index, #waypoints do
if waypoints[i].rev and not waypoints[i-1].wait then
tcx = getX(waypoints, i);
tcz = getZ(waypoints, i);
else
local dx, dz, _ = courseplay:getPointDirection(waypoints[i-2], waypoints[i-1]);
tcx = getX(waypoints, i-1) + dx * (waypoints[i-1].wait and 15 or 30);
tcz = getZ(waypoints, i-1) + dz * (waypoints[i-1].wait and 15 or 30);
end;
local distance = courseplay:distance(xTipper,zTipper, getX(waypoints, i-1) ,getZ(waypoints, i-1));
local waitingPoint;
local unloadPoint;
if waypoints[i-1].wait then
waitingPoint = i-1;
end;
if waypoints[i].wait then
waitingPoint = i;
end;
if waypoints[i-1].unload then
unloadPoint = i-1;
end;
if waypoints[i].unload then
unloadPoint = i;
end;
-- HANDLE WAITING POINT WAYPOINT CHANGE
if waitingPoint then
if workTool.cp.realUnloadOrFillNode then
local _,y,_ = getWorldTranslation(workTool.cp.realUnloadOrFillNode);
local _,_,z = worldToLocal(workTool.cp.realUnloadOrFillNode, getX(waypoints, waitingPoint), y, getZ(waypoints, waitingPoint));
if z >= 0 then
courseplay:setWaypointIndex(vehicle, waitingPoint + 1);
courseplay:debug(string.format("%s: Is at waiting point", nameNum(vehicle)), courseplay.DBG_REVERSE);
break;
end;
else
if distance <= 2 then
courseplay:setWaypointIndex(vehicle, waitingPoint + 1);
courseplay:debug(string.format("%s: Is at waiting point", nameNum(vehicle)), courseplay.DBG_REVERSE);
break;
end;
end;
if distance > 3 then
local _,_,z = worldToLocal(node, tcx,yTipper,tcz);
if z < 0 then
courseplay:setWaypointIndex(vehicle, i - 1);
break;
end;
end;
break;
elseif unloadPoint then
if workTool.cp.rearTipRefPoint then
local tipRefPoint = workTool.tipReferencePoints[workTool.cp.rearTipRefPoint].node
local x,y,z = getWorldTranslation(tipRefPoint);
local tipDistanceToPoint = courseplay:distance(x,z,getX(waypoints, unloadPoint),getZ(waypoints, unloadPoint))
courseplay:debug(string.format("%s:workTool.cp.rearTipRefPoint: tipDistanceToPoint: %s", nameNum(vehicle),tostring(tipDistanceToPoint)), courseplay.DBG_REVERSE);
if tipDistanceToPoint < 0.5 then
courseplay:setWaypointIndex(vehicle, unloadPoint + 1);
courseplay:debug(string.format("%s: Is at unload point", nameNum(vehicle)), courseplay.DBG_REVERSE);
break;
end;
end
if distance > 3 then
local _,_,z = worldToLocal(node, tcx,yTipper,tcz);
if z < 0 then
courseplay:setWaypointIndex(vehicle, i - 1);
break;
end;
end;
break
-- SWITCH TO FORWARD
elseif waypoints[i-1].rev and not waypoints[i].rev then
if distance <= 2 then
courseplay:debug(string.format("%s: Change direction to forward", nameNum(vehicle)), courseplay.DBG_REVERSE);
end;
break;
-- FIND THE RIGHT START REVERSING WAYPOINT
elseif waypoints[i-1].rev and not waypoints[i-2].rev then
for recNum = index, vehicle.cp.numWaypoints do
local srX,srZ = getX(waypoints, recNum),getZ(waypoints, recNum);
local _,_,tsrZ = worldToLocal(node,srX,yTipper,srZ);
if tsrZ < -2 then
courseplay:setWaypointIndex(vehicle, recNum);
courseplay:debug(string.format("%s: First reverse point -> Change waypoint to behind trailer: %q", nameNum(vehicle), recNum), courseplay.DBG_REVERSE);
break;
end;
end;
break;
-- HANDLE REVERSE WAYPOINT CHANGE
elseif distance > 3 then
local _,_,z = worldToLocal(node, tcx,yTipper,tcz);
if z < 0 then
courseplay:setWaypointIndex(vehicle, i - 1);
break;
end;
end;
end;
elseif mode2 then
tcx,tcz = vehicle.cp.curTarget.x, vehicle.cp.curTarget.z;
end;
if debugActive then
cpDebug:drawPoint(tcx, yTipper+3, tcz, 1, 1 , 1)
if workTool.cp.realUnloadOrFillNode then
local xUOFNode,yUOFNode,zUOFNode = getWorldTranslation(workTool.cp.realUnloadOrFillNode);
cpDebug:drawPoint(xUOFNode,yUOFNode+5,zUOFNode, 0, 1 , 0.5);
end;
end;
local lxTipper, lzTipper = AIVehicleUtil.getDriveDirection(node, tcx, yTipper, tcz);
courseplay:showDirection(node,lxTipper, lzTipper, 1, 0, 0);
local lxFrontNode, lzFrontNode = AIVehicleUtil.getDriveDirection(frontNode, xTipper,yTipper,zTipper);
local lxTractor, lzTractor = 0,0;
local maxTractorAngle = rad(60);
-- for articulated vehicles use the articulated axis' rotation node as it is a better indicator or the
-- vehicle's orientation than the direction node which often turns/moves with an articulated vehicle part
-- TODO: consolidate this with AITurn:getTurnNode()
local turnNode
local useArticulatedAxisRotationNode = SpecializationUtil.hasSpecialization(ArticulatedAxis, vehicle.specializations) and vehicle.spec_articulatedAxis.rotationNode
if useArticulatedAxisRotationNode then
turnNode = vehicle.spec_articulatedAxis.rotationNode
else
turnNode = vehicle.cp.directionNode
end
if workTool.cp.isPivot then
courseplay:showDirection(frontNode,lxFrontNode, lzFrontNode, 0, 1, 0);
lxTractor, lzTractor = AIVehicleUtil.getDriveDirection(turnNode, xFrontNode,yFrontNode,zFrontNode);
courseplay:showDirection(turnNode,lxTractor, lzTractor, 0, 0.7, 0);
local rotDelta = (workTool.cp.nodeDistance * (0.5 - (0.023 * workTool.cp.nodeDistance - 0.073)));
local trailerToWaypointAngle = courseplay:getLocalYRotationToPoint(node, tcx, yTipper, tcz, -1) * rotDelta;
trailerToWaypointAngle = MathUtil.clamp(trailerToWaypointAngle, -rad(90), rad(90));
local dollyToTrailerAngle = courseplay:getLocalYRotationToPoint(frontNode, xTipper, yTipper, zTipper, -1);
local tractorToDollyAngle = courseplay:getLocalYRotationToPoint(turnNode, xFrontNode, yFrontNode, zFrontNode, -1);
local rearAngleDiff = (dollyToTrailerAngle - trailerToWaypointAngle);
rearAngleDiff = MathUtil.clamp(rearAngleDiff, -rad(45), rad(45));
local frontAngleDiff = (tractorToDollyAngle - dollyToTrailerAngle);
frontAngleDiff = MathUtil.clamp(frontAngleDiff, -rad(45), rad(45));
local angleDiff = (frontAngleDiff - rearAngleDiff) * (1.5 - (workTool.cp.nodeDistance * 0.4 - 0.9) + rotDelta);
angleDiff = MathUtil.clamp(angleDiff, -rad(45), rad(45));
lx, lz = MathUtil.getDirectionFromYRotation(angleDiff);
else
lxTractor, lzTractor = AIVehicleUtil.getDriveDirection(turnNode, xTipper,yTipper,zTipper);
courseplay:showDirection(turnNode,lxTractor, lzTractor, 1, 1, 0);
local rotDelta = workTool.cp.nodeDistance * 0.3;
local trailerToWaypointAngle = courseplay:getLocalYRotationToPoint(node, tcx, yTipper, tcz, -1) * rotDelta;
trailerToWaypointAngle = MathUtil.clamp(trailerToWaypointAngle, -math.rad(90), math.rad(90));
local tractorToTrailerAngle = courseplay:getLocalYRotationToPoint(turnNode, xTipper, yTipper, zTipper, -1);
local angleDiff = (tractorToTrailerAngle - trailerToWaypointAngle) * (1 + rotDelta);
-- If we only have steering axle on the worktool and they turn when reversing, we need to steer a lot more to counter this.
if workTool.cp.steeringAxleUpdateBackwards then
angleDiff = angleDiff * 4;
end;
angleDiff = MathUtil.clamp(angleDiff, -maxTractorAngle, maxTractorAngle);
lx, lz = MathUtil.getDirectionFromYRotation(angleDiff);
end;
if (mode == courseplay.MODE_GRAIN_TRANSPORT or mode == courseplay.MODE_COMBI or mode == courseplay.MODE_FIELDWORK) and vehicle.cp.currentTipTrigger == nil and (vehicle.cp.totalFillLevel ~= nil and vehicle.cp.totalFillLevel > 0) then
local nx, ny, nz = localDirectionToWorld(node, lxTipper, -0.1, lzTipper);
courseplay:doTriggerRaycasts(vehicle, 'tipTrigger', 'rev', false, xTipper, yTipper + 1, zTipper, nx, ny, nz);
end;
courseplay:showDirection(turnNode,lx,lz, 0.7, 0, 1);
-- do a little bit of damping if using the articulated axis as lx tends to oscillate around 0 which results in the
-- speed adjustment kicking in and slowing down the vehicle.
if useArticulatedAxisRotationNode and math.abs(lx) < 0.04 then lx = 0 end
-- true means this code is taking care of the reversing as this is not a trivial case
-- for instance because of a trailer
return lx,lz,fwd, true;
end;
function courseplay:getFirstReversingWheeledWorkTool(vehicle)
-- since some weird things like Seed Bigbag are also vehicles, check this first
if not vehicle.getAttachedImplements then return nil end
-- Check all attached implements if we are an wheeled workTool behind the tractor
for _, imp in ipairs(vehicle:getAttachedImplements()) do
-- Check if the implement is behind
if courseplay:isRearAttached(vehicle, imp.jointDescIndex) then
if courseplay:isWheeledWorkTool(imp.object) then
-- If the implement is a wheeled workTool, then return the object
return imp.object;
else
-- If the implement is not a wheeled workTool, then check if that implement have an attached wheeled workTool and return that.
return courseplay:getFirstReversingWheeledWorkTool(imp.object);
end;
end;
end;
-- If we didnt find any workTool, return nil
return nil;
end;
function courseplay:getLocalYRotationToPoint(node, x, y, z, direction)
direction = direction or 1;
local dx, _, dz = worldToLocal(node, x, y, z);
dx = dx * direction;
dz = dz * direction;
return MathUtil.getYRotationFromDirection(dx, dz);
end;
function courseplay:showDirection(node,lx,lz, r, g, b)
if courseplay.debugChannels[courseplay.DBG_REVERSE] then
local x,y,z = getWorldTranslation(node);
local ctx,_,ctz = localToWorld(node,lx*5,y,lz*5);
cpDebug:drawLine(x, y+5, z, r or 1, g or 0, b or 0, ctx, y+5, ctz);
end
end
-- Find the first forward waypoint ahead of the vehicle
function courseplay:getNextFwdPoint(vehicle, isTurning)
local directionNode = AIDriverUtil.getDirectionNode(vehicle)
if isTurning then
courseplay:debug(('%s: getNextFwdPoint()'):format(nameNum(vehicle)), courseplay.DBG_REVERSE);
-- scan only the next few waypoints, we don't want to end up way further in the course, missing
-- many waypoints. The proper solution here would be to take the workarea into account as the tractor may
-- be well ahead of the turnEnd point in case of long implements. Instead we just assume 10 waypoints is
-- long enough.
for i = vehicle.cp.waypointIndex, math.min( vehicle.cp.waypointIndex + 10, vehicle.cp.numWaypoints ) do
local waypointToCheck = vehicle.Waypoints[i]
if not waypointToCheck.rev and not waypointToCheck.turnEnd then
local wpX, wpZ = waypointToCheck.cx, waypointToCheck.cz;
local _, _, disZ = worldToLocal(directionNode, wpX, getTerrainHeightAtWorldPos(g_currentMission.terrainRootNode, wpX, 300, wpZ), wpZ);
local vX, _, vZ = localToWorld( directionNode, 0, 0, 0 )
courseplay:debug(('%s: getNextFwdPoint(), vX = %.1f, vZ = %.1f, i = %d, wpX = %.1f, wpZ = %.1f, disZ = %.1f '):format(
nameNum(vehicle), vX, vZ, i, wpX, wpZ, disZ ), courseplay.DBG_REVERSE);
if disZ > 5 then
courseplay:debug(('--> return (%d) as waypointIndex'):format(i), courseplay.DBG_REVERSE);
return i;
end;
end;
end;
local ix = math.min(vehicle.cp.waypointIndex + 1, vehicle.cp.numWaypoints)
courseplay:debug(('\tno waypoint found in front of us, returning next waypoint (%d)'):format(ix), courseplay.DBG_REVERSE);
return ix
else
local maxVarianceX = sin(rad(30));
local firstFwd, firstFwdOver3;
courseplay:debug(('%s: getNextFwdPoint()'):format(nameNum(vehicle)), courseplay.DBG_REVERSE);
for i = vehicle.cp.waypointIndex, vehicle.cp.numWaypoints do
if not vehicle.Waypoints[i].rev then
local x, y, z = getWorldTranslation(directionNode);
local wdx, _, wdz, dist = courseplay:getWorldDirection(x, 0, z, vehicle.Waypoints[i].cx, 0, vehicle.Waypoints[i].cz);
local dx,_,dz = worldDirectionToLocal(directionNode, wdx, 0, wdz);
if not firstFwd then
firstFwd = i;
courseplay:debug(('--> set firstFwd as %d'):format(i), courseplay.DBG_REVERSE);
end;
if not firstFwdOver3 and dz and dist and dz * dist >= 3 then
firstFwdOver3 = i;
courseplay:debug(('--> set firstFwdOver3 as %d'):format(i), courseplay.DBG_REVERSE);
end;
courseplay:debug(('--> point %d, dx=%.4f, dz=%.4f, dist=%.2f, maxVarianceX=%.4f'):format(i, dx, dz, dist, maxVarianceX), courseplay.DBG_REVERSE);
if dz > 0 and abs(dx) <= maxVarianceX then -- forward and x angle <= 30°
courseplay:debug('----> return as waypointIndex', courseplay.DBG_REVERSE);
return i;
end;
end;
end;
if firstFwdOver3 then
courseplay:debug(('\treturn firstFwdOver3 (%d)'):format(firstFwdOver3), courseplay.DBG_REVERSE);
return firstFwdOver3;
elseif firstFwd then
courseplay:debug(('\treturn firstFwd (%d)'):format(firstFwd), courseplay.DBG_REVERSE);
return firstFwd;
end;
end;
courseplay:debug('\treturn 1', courseplay.DBG_REVERSE);
return 1;
end;
function courseplay:getReverseProperties(vehicle, workTool)
courseplay:debug(('getReverseProperties(%q, %q)'):format(nameNum(vehicle), nameNum(workTool)), courseplay.DBG_REVERSE);
-- Make sure they are reset so they wont conflict when changing worktools
workTool.cp.frontNode = nil;
workTool.cp.isPivot = nil;
if workTool == vehicle then
courseplay:debug('--> workTool is vehicle (steerable) -> return', courseplay.DBG_REVERSE);
return;
end;
if vehicle.cp.hasSpecializationShovel then
courseplay:debug('--> vehicle has "Shovel" spec -> return', courseplay.DBG_REVERSE);
return;
end;
-- Sugarcane trailer for some reason also have a shovel specialization but we still want to use
-- them as trailers and reverse them, for which we need the realTurningNode which is set below.
-- So only ignore tools which do have a shovel but no wheel...
if workTool.cp.hasSpecializationShovel and not courseplay:isWheeledWorkTool(workTool) then
courseplay:debug('--> workTool has "Shovel" spec and has no wheels -> return', courseplay.DBG_REVERSE);
return;
end;
if not courseplay:isWheeledWorkTool(workTool) or courseplay:isHookLift(workTool) or courseplay:isAttacherModule(workTool) then
courseplay:debug('--> workTool doesn\'t need reverse properties -> return', courseplay.DBG_REVERSE);
return;
end;
--------------------------------------------------
local attacherVehicle = workTool:getAttacherVehicle();
if not workTool.cp.distances then
workTool.cp.distances = courseplay:getDistances(workTool);
end;
-- TODO: figure out why these are only set for reversing, one would think they are pretty generic
-- and should be set in all cases...
workTool.cp.realTurningNode = courseplay:getRealTurningNode(workTool);
workTool.cp.realUnloadOrFillNode = courseplay:getRealUnloadOrFillNode(workTool);
if attacherVehicle == vehicle or attacherVehicle.cp.isAttacherModule then
workTool.cp.frontNode = courseplay:getRealTrailerFrontNode(workTool);
else
workTool.cp.frontNode = courseplay:getRealDollyFrontNode(attacherVehicle);
if workTool.cp.frontNode then
courseplay:debug(string.format('--> workTool %q has dolly', nameNum(workTool)), courseplay.DBG_REVERSE);
else
courseplay:debug(string.format('--> workTool %q has invalid dolly -> return', nameNum(workTool)), courseplay.DBG_REVERSE);
return;
end;
end;
workTool.cp.nodeDistance = courseplay:getRealTrailerDistanceToPivot(workTool);
courseplay:debug("--> tz: "..tostring(workTool.cp.nodeDistance).." workTool.cp.realTurningNode: "..tostring(workTool.cp.realTurningNode), courseplay.DBG_REVERSE);
if workTool.cp.realTurningNode == workTool.cp.frontNode then
courseplay:debug('--> workTool.cp.realTurningNode == workTool.cp.frontNode', courseplay.DBG_REVERSE);
workTool.cp.isPivot = false;
else
workTool.cp.isPivot = true;
end;
courseplay:debug(('--> isPivot=%s, frontNode=%s'):format(tostring(workTool.cp.isPivot), tostring(workTool.cp.frontNode)), courseplay.DBG_REVERSE);
end;