Butterfly is all about WaveSurfing. It's the result of letting the Pugilist movement grow out of its MiniBot constraints. The design is pretty simple:
Simple enough, huh? One of those steps is a bit complicated though - step 2.1. It involves predicting our own movement some ticks in the future. This can be done in many ways. Pugilist does it in a pretty sloppy manner. CassiusClay is a bit more precise. In fact as of version 1.9.9.00g (g for gamma, my first test version) not only considers how fast CC can accelerate/deccelerate. It also tries to take into account the maximum turn angle it can do at a given velocity. Here's the scheme for the predictor:
Inputs
Action
Here's the implementing code (RWPCL):
Move waveImpactLocation(MovementWave closest, double direction, double maxVelocity) { double currentDirection = robotOrbitDirection(closest.gunBearing(robotLocation)); double v = Math.abs(robot.getVelocity()) * PUtils.sign(direction); double h = robot.getHeadingRadians(); Point2D orbitCenter = orbitCenter(closest); Point2D impactLocation = new Point2D.Double(robot.getX(), robot.getY()); Move smoothed; int time = 0; do { smoothed = wallSmoothedDestination(impactLocation, orbitCenter, currentDirection * direction); double wantedHeading = PUtils.absoluteBearing(impactLocation, smoothed.location); h += PUtils.backAsFrontDirection(wantedHeading, h) < 0 ? Math.PI : 0.0; if (v < maxVelocity) { v = Math.min(maxVelocity, v + (v < 0 ? 2 : 1)); } else { v = Math.max(maxVelocity, v - 2); } double maxTurn = Math.toRadians(MAX_TURN_RATE - 0.75 * Math.abs(v)); h += PUtils.minMax(PUtils.backAsFrontTurn(wantedHeading, h), -maxTurn, maxTurn); impactLocation = PUtils.project(impactLocation, h, v); } while (closest.distanceFromTarget(impactLocation, time++) > 18); return new Move(impactLocation, smoothed.smoothing, smoothed.wantedEvasion, smoothed.oldDistance, impactLocation.distance(enemyLocation)); }
The BackAsFront functions there looks like so:
public static double backAsFrontTurn(double newHeading, double oldHeading) { return Math.tan(newHeading - oldHeading); } public static double backAsFrontDirection(double newHeading, double oldHeading) { return sign(Math.cos(newHeading - oldHeading)); }
It's the same functions I use to move my bot around Jamougha style:
double newHeading = PUtils.absoluteBearing(robotLocation, destination); double oldHeading = robot.getHeadingRadians(); robot.setAhead(PUtils.backAsFrontDirection(newHeading, oldHeading) * 50); robot.setTurnRightRadians(PUtils.backAsFrontTurn(newHeading, oldHeading));
-- PEZ
Thanks! I did something like that actually, though not in GL. Maybe it wasn't clear, but I have verified now that the above code does work for its purpose. So there's no problem. But I'll let the page be published. Maybe someone can learn something from it. Since you might still be wondering, here's the wall smoothing function:
static Move wallSmoothedDestination(Point2D location, Point2D orbitLocation, double direction) { Point2D destination = new Point2D.Double(); destination.setLocation(location); double distance = enemyLocation.distance(location); double evasion = evasion(distance); double blindStick = enemyIsRammer() ? PUtils.minMax(enemyDistance / 2, 60, 170) : 150; double smoothing = 0; while (!fieldRectangle.contains(destination = PUtils.project(location, PUtils.absoluteBearing(location, orbitLocation) - direction * ((evasion - smoothing / 100) * Math.PI / 2), blindStick)) && smoothing < MAX_WALL_SMOOTH_TRIES) { smoothing++; } return new Move(destination, smoothing, evasion, distance, destination.distance(enemyLocation)); }Yes, much the same as RandomMovementBot is using.
-- PEZ
Ok, thanx. Surprisingly simple solution! --Vic
Maybe so. But I'm very proud of it! =) It started the whole Tityus - Raiko race and was a major key to create two really competetive micros (RaikoMicro and Aristocles). -- PEZ
Don't get me wrong, that was a compliment! Simple==good (when it works:) --Vic
I didn't get you wrong. See that smiley? I think most of us like simplicity in key parts of our bots. It gives room for complexity elsewhere if nothing else. -- PEZ
Why are you firing enemy waves every tick? Isn't it enough to just fire 'surfable' waves? --Vic
For the purposes of the above described scheme, yes. But part of the story is not told. =) I have a flattening "under current" in my surfing. And that surfing uses visit counts from those every-tick waves. Furthermore, many of my methods and functions need a wave to work with and since i don't always have a fresh surfable wave at hand i hand them the latest created wave, regardless if it's surfabe or not. -- PEZ
Great explanation! May be I'w even try a bot with WaveSurfing. BTW, to calculate your exact future position, you can use FuturePosition. -- Albert
Thanks! I was hoping it would help make WaveSurfing a bit less voodoo-ish. Funny that the author of FuturePosition would be someone finding the explanation useful. =) Note that this is just one way to go about wave surfing. Axe is doing something different. Which he eventualy will explain on the SilverSurfer/WaveSuffering page. -- PEZ
if (velocityIndex != lastVelocityIndex) { timeSinceVChange = 0; wave.accelIndex = velocityIndex < lastVelocityIndex ? 1 : 2; } wave.accelIndex = 0;shouldn't it be
wave.accelIndex = 0; if (velocityIndex != lastVelocityIndex) { timeSinceVChange = 0; wave.accelIndex = velocityIndex < lastVelocityIndex ? 1 : 2; }Florent
Florent, it looks like you must be right about that, so I figured I'd test it in the CurveFlatteningChallenge both ways. The score was actually higher with it the way it is, so maybe he did that to just disable the accelIndex? It was 96.52% vs 97.32%. A few matches vs Ascendant all gave CC a higher score percentage than the 40.8% he has now in the RR, though, so maybe it really is a mistake! I'm tempted to make a new bot that is CC with that one thing changed and enter it in the RoboRumble to see what happens... ;) -- Voidious
Hey PEZ, you probably never saw this comment, but now that you're checking the wiki again... This certainly seems like it could possibly be a bug in CassiusClay's movement, unless you intended to turn off the accel index. -- Voidious
Wow, that sure looks stupid, doesn't it? It's hard to remember, but I know I wouldn't switch off accelIndex like that. I checked the code now and I saw that the accelIndex isn't used in the segmentation. I think that the Florent-bug made accelIndex useless and thus I probably tuned it away. Now, wouldn't it be a bit interesting to tune CC for using it again now that Florent showed how to enable it? This is prolly extremely belated Florent, but HUGE thanks! And thanks to Voidious for picking it up and for reminding me now that I am paying attention again.
The changes in performance you saw there, V, must have been chance in play. That's how inexact the indexes are I guess.
-- PEZ
I was checking my own math, comparing with yours when I found that:
approachVelocity = velocity * -Math.cos(robot.getHeadingRadians() - (enemyAbsoluteBearing + 180));That's line 114, I think adding degrees (180) and radians is not a good idea. -- Florent
Wow! And thanks! Even if belated, voidious mentioned this bug report to me yesterday. I'll release a bugfixed version in a jiffy. -- PEZ
Seems I gained 2 points with that bug fix. It might or migt not be a significant change, but at least I think it can be said that fixing this bug didn't make things worse. =) And now that this bug is fixed I might be able to tweak the weights of the segmentation to squeeze out some more points. -- PEZ
Finally somebody came up with the solution