package pfvicm;
import robocode.*;
import java.awt.Color;
import java.awt.geom.*;
import robocode.util.Utils;
/**
 * Suvorov - PFVICM 8.0.2
 */
public class Suvorov extends AdvancedRobot
{
	static final double PI = Math.PI;
	public double absbearing;
	static int FORWARD = 1;
	public double firePower;
	public double changehead, y, time, heading, speed, x, starthead, nexthead;
	public double tx, ty;
	public void run() {
		setColors(Color.cyan,Color.green,Color.red);
		setAdjustGunForRobotTurn(true);
		setAdjustRadarForGunTurn(true);
		do {
			turnRadarLeft(Double.POSITIVE_INFINITY);
		}
		while(true);
}
	public void onScannedRobot(ScannedRobotEvent e) {
		double distance = Math.random()*(Math.max(Math.max((getBattleFieldHeight()-getY()-20),(getBattleFieldWidth()-getX()-20)),(Math.max(getX(),getY()))));
		if (getDistanceRemaining() == 0) { FORWARD = -FORWARD; setAhead(distance * FORWARD); }
		setTurnRightRadians(e.getBearingRadians() + PI/2 - 0.5236 * FORWARD * (e.getDistance() > 200 ? 1 : -1));
		tx = getX()+Math.sin(absbearing)*e.getDistance();
     	ty = getY()+Math.cos(absbearing)*e.getDistance();
		speed = Math.abs(getVelocity());
		starthead = getHeading();
		nexthead = getHeading();
		firePower = Math.min(3.0,getEnergy()/2);
		doGun();
		fire(firePower);
	}	
	public Point2D.Double guessPosition(double when) {
    double diff = when - time;
    double newX, newY;
	double changehead = Math.abs(starthead - nexthead);
    if (Math.abs(changehead) > 0.00001) {
        double radius = speed/changehead;
        double tothead = diff * changehead;
        newY = ty + (Math.sin(heading + tothead) * radius) - 
                      (Math.sin(heading) * radius);
        newX = tx + (Math.cos(heading) * radius) - 
                      (Math.cos(heading + tothead) * radius);
    }
    else {
        newY = ty + Math.cos(heading) * speed * diff;
        newX = tx + Math.sin(heading) * speed * diff;
    }
    return new Point2D.Double(newX, newY);
}
	void doGun() {
    	double time;
    	double nextTime;
    	Point2D.Double p;
    	p = new Point2D.Double(tx, ty);
    	for (int i = 0; i < 10; i++){
     	   nextTime = 
    	Math.round((getRange(getX(),getY(),p.x,p.y)/(20-(3*firePower))));
        	time = getTime() + nextTime;
       		p = guessPosition(time);
    	}
    double gunOffset = getGunHeadingRadians() - 
            (PI/2 - Math.atan2(p.y - getY(), p.x - getX()));
   		 setTurnGunLeftRadians(normaliseBearing(gunOffset));
	}
	double normaliseBearing(double ang) {
    	if (ang > PI)
        	ang -= 2*PI;
    	if (ang < -PI)
    	    ang += 2*PI;
    	return ang;
	}
	public double getRange(double x1,double y1, double x2,double y2) {
    	double x = x2-x1;
    	double y = y2-y1;
    	double h = Math.sqrt( x*x + y*y );
    	return h;	
	}
	public double absbearing(Point2D source, Point2D target) {
		return Math.toDegrees(Math.atan2(target.getX() - source.getX(), target.getY() - source.getY()));
	}
}
Questions/Comments?:
Well, you must have got rid of to much stuff to get it to compile. None of x, y, speed, heading, absbearing, changehead is given any value. Also you might want change in guessPosition x and y to tx and ty. -- Florent
I think the problem is that you never set tx and ty properly in the scannedRobot method. In your scannedRobot method you have:
 double tx = getX()+Math.sin(absbearing)*e.getDistance();
      double ty = getY()+Math.cos(absbearing)*e.getDistance();
those values are only used within the scope of that method. I think you want to set the global variables tx, ty (which are later used in doGun()).
just replace the above lines with:
tx = getX()+Math.sin(absbearing)*e.getDistance();
     ty = getY()+Math.cos(absbearing)*e.getDistance();
There may be more problems, but that jumps out at me.
EDIT: I just looked through that code more carefully and there are definitely more problems. Florent is right, all of those values need to be set in order for this to work.
--wcsv
Code above updated to current. Still doesn't work, fires due north. --Pfvicm
One quick remark: You never give absbearing a value, so it is always 0. In your scannedrobotevent it should be assigned the value getHeadingRadians() + e.getBearingRadians?() . If you just need a circular targeting routine, take a look at WaveSurfingChallengeBotC. -- GrubbmGait
Thanks to all. Problem is not yet solved - he fires at angles between 0 and 90 - but I will inspect Bot C and and see what's missing from my routine. -- Pfvicm