//TODO: //Auto //Sonar code #include "WPILib.h" #include "SmartDashboard/SmartDashboard.h" #include "Command.h" #include #include #include #include class RobotDemo : public SimpleRobot { RobotDrive myRobot; float potVal, multiplier, servoXState, servoYState, throttle, ServoXJoyPos, ServoYJoyPos; int lastToggle; int collectorSpeed; bool collectorExtended, toggleCollector; bool shooting, compressing; float DownSpeed, downLimit, upLimit; string cmd; Joystick Rstick, Lstick; Servo Servo1, Servo2; Solenoid collectorSole1, collectorSole2; DigitalInput pneumaticLS, shootTopLS, shootBottomLS; Relay collectorSpike, lightingSpike; Compressor compressor; Jaguar Left1, Left2, Left3, Right1, Right2, Right3, RightArmMotor1, RightArmMotor2, LeftArmMotor1, LeftArmMotor2, CollectorMotor1; AnalogChannel armPot; //Ultrasonic AnalogChannel BallSonicLeft, BallSonicRight, WallSonicLeft, WallSonicRight; DigitalOutput BallLeft, BallRight, WallLeft, WallRight; public: RobotDemo(): //Joysticks Rstick(1), Lstick(2), //Pot armPot(6), //Ultrasonic BallSonicLeft(1), BallSonicRight(2), WallSonicLeft(3), WallSonicRight(4), BallLeft(1,4), WallLeft(1,5), BallRight(2,4), WallRight(2,5), //Compressor //compressor(2,3,1,1), compressor(2, 5, 1, 1), //Solenoids collectorSole1(1), collectorSole2(2), //Limit switches pneumaticLS(1), shootTopLS(2), shootBottomLS(3), //Driver Motors Left1(1, 1), Left2(1, 2), Left3(1, 3), Right1(2, 1), Right2(2, 2), Right3(2, 3), //Servos Servo1(1, 7), Servo2(1, 8), //Spikes collectorSpike(2, 7), lightingSpike(2, 8), //Shooter Motors LeftArmMotor1(1, 4), LeftArmMotor2(1, 5), RightArmMotor1(2, 4), RightArmMotor2(2, 5), //Collector Motor CollectorMotor1(1, 6), myRobot(Left1, Left2, Right1, Right2) { GetWatchdog().SetEnabled(false); } void RobotInit() { //Initializing robot lastToggle = 0; DashboardSetup(); servoXState = 90; servoYState = 90; multiplier = 1.0f; downLimit = 40; upLimit = 130; compressor.Start(); shooting = false; compressing = true; toggleCollector = false; throttle=0; } void DashboardSetup() { SmartDashboard::PutNumber("Throttle", throttle); SmartDashboard::PutNumber("downLimit", downLimit); SmartDashboard::PutNumber("upLimit", upLimit); SmartDashboard::PutNumber("DownSpeed", 0.100); SmartDashboard::PutString("Auto", cmd); SmartDashboard::PutNumber("collectorSpeed", 127); SmartDashboard::PutNumber("armPot", potToDegrees(armPot.GetAverageVoltage())); SmartDashboard::PutNumber("Log Level", 1); //Ultrasonic SmartDashboard::PutNumber("Wall Left", voltToDistance(WallSonicLeft.GetAverageVoltage(),true)); SmartDashboard::PutNumber("Wall Right", voltToDistance(WallSonicRight.GetAverageVoltage(),true)); SmartDashboard::PutNumber("Ball Left", voltToDistance(BallSonicLeft.GetAverageVoltage())); SmartDashboard::PutNumber("Ball Right", voltToDistance(BallSonicRight.GetAverageVoltage())); SmartDashboard::PutBoolean("Use ultrasonic",false); } void updateDashboard() { SmartDashboard::PutNumber("Throttle", throttle); collectorSpeed = SmartDashboard::GetNumber("collectorSpeed"); SmartDashboard::PutNumber("armPot", potToDegrees(armPot.GetAverageVoltage())); //Ultrasonic SmartDashboard::PutNumber("Wall Left", voltToDistance(WallSonicLeft.GetAverageVoltage(),true)); SmartDashboard::PutNumber("Wall Right", voltToDistance(WallSonicRight.GetAverageVoltage(),true)); SmartDashboard::PutNumber("Ball Left", voltToDistance(BallSonicLeft.GetAverageVoltage())); SmartDashboard::PutNumber("Ball Right", voltToDistance(BallSonicRight.GetAverageVoltage())); DownSpeed = SmartDashboard::GetNumber("DownSpeed"); downLimit = SmartDashboard::GetNumber("downLimit"); upLimit = SmartDashboard::GetNumber("upLimit"); if (downLimit < 35) { downLimit = 35; } if (upLimit > 167) { upLimit = 167; } } void shootRobot(float power=0, bool override=false) { if(!SmartDashboard::GetBoolean("Use ultrasonic")){ override=true; } //Needs a limit to help the driver aim //In this case its checking that we are no more than 15 degrees off //The override is in place in case an ultrasonic becomes damaged and we are unable to validate the distance through software float averageAtan = atan((abs(voltToDistance(BallSonicLeft.GetAverageVoltage()) - voltToDistance(BallSonicRight.GetAverageVoltage())))/20.0f)*360.0f/(2.0f*3.141592653589793f); if(override==true){ printf("%f\n",power); setMotorValue(4, 1, cvt(power)); setMotorValue(5, 1, cvt(power)); setMotorValue(4, 2, cvt(-power)); setMotorValue(5, 2, cvt(-power)); }else if(averageAtan<=30.0f&&(voltToDistance(WallSonicLeft.GetAverageVoltage(),true))<=100&&(voltToDistance(WallSonicRight.GetAverageVoltage(),true))<=100){ setMotorValue(4, 1, cvt(power)); setMotorValue(5, 1, cvt(power)); setMotorValue(4, 2, cvt(-power)); setMotorValue(5, 2, cvt(-power)); } if(averageAtan>=16.0f&&(voltToDistance(WallSonicLeft.GetAverageVoltage(),true))<=100&&(voltToDistance(WallSonicRight.GetAverageVoltage(),true))){ //printf/smartdashboard: warning setMotorValue(4, 1, cvt(power)); setMotorValue(5, 1, cvt(power)); setMotorValue(4, 2, cvt(-power)); setMotorValue(5, 2, cvt(-power)); } //Allow shooting regardless of the angle or distance } void logMsg(std::string message, int level){ if((int)SmartDashboard::GetNumber("Log Level") % level == 0){ printf((message+"\n").c_str()); } } void driveRobot(float x, float y) { if(y>1.0f){ y=1.0f; }else if(y!=0.0f&&y<-1.0f){ y=-1.0f; } int leftPower = ((y+x)/2+1)*127+1; int rightPower = ((y-x)/2+1)*127+1; //logMsg("leftPower: "+toString(leftPower),3); //logMsg("rightPower: "+toString(rightPower),3); //logMsg("JoyX: "+toString(Rstick.GetX()),3); //logMsg("JoyY: "+toString(Rstick.GetY()),3); setMotorValue(1, 1, leftPower); setMotorValue(2, 1, leftPower); setMotorValue(3, 1, leftPower); setMotorValue(1, 2, rightPower); setMotorValue(2, 2, rightPower); setMotorValue(3, 2, rightPower); } template string toString(numbertype a){ //TODO stringstream ss; ss<38){ WallLeft.Set(0); BallLeft.Set(0); WallRight.Set(1); BallRight.Set(1); avgRight+=voltToDistance(WallSonicRight.GetAverageVoltage()); //After 50 cur iterations, find the average (data was recorded 12 times, divide by 12) }else if(cur==50){ cur=0; avgRight/=12; thisIsATest=atan((abs(avgLeft-avgRight))/20.0f)*360.0f/(2.0f*3.141592653589793f); SmartDashboard::PutNumber("Tanval", atan((abs(avgLeft-avgRight))/20.0f)*360.0f/(2.0f*3.141592653589793f)); } //Calculate the inital distance and average it averageAmount times. if(i=5&&avgDist-curDist<=36.0f) { float xPower, yPower; xPower=1; yPower=(avgDist-curDist)/36.0f; if(yPower>1){ yPower=1; } driveRobot(yPower, xPower); setMotorValue(6, 1, 1); } else if (i>1400&&i<1600&&125>=potToDegrees(armPot.GetAverageVoltage())) { driveRobot(0, 0); shootRobot(1, false); setMotorValue(6, 1, 1); } else if (i>1500&&i<1700) { driveRobot(-1,0); } else { driveRobot(0, 0); shootRobot(0, true); setMotorValue(6, 1, 0); } updateDashboard(); if (i % 100 == 0 && compressing && compressor.GetPressureSwitchValue() == 1) { compressor.Stop(); compressing = false; logMsg("Stopping the compressor",2); } if (i % 100 == 0 && !compressing && compressor.GetPressureSwitchValue() == 0) { compressor.Start(); compressing = true; logMsg("Starting the compressor",2); } Wait(0.005f); i++; cur++; } compressing = false; compressor.Stop(); } void OperatorControl() { myRobot.SetSafetyEnabled(false); int i = 0; int cur=0; bool swap=false; collectorSole1.Set(true); collectorSole2.Set(false); compressing = false; logMsg("Starting Teleop",1); while (IsEnabled() && IsOperatorControl()) { if(cur==50){ cur=0; WallLeft.Set(swap?1:0); BallRight.Set(swap?1:0); WallRight.Set(swap?0:1); BallRight.Set(swap?0:1); swap=!swap; } throttle = (-Lstick.GetRawAxis(4)+1)/2; driveRobot(Rstick.GetY(),Rstick.GetZ()+Rstick.GetX()); //Log things if (i % 200 == 0) { //logMsg(toString(compressor.GetPressureSwitchValue()),2); //logMsg("armPot value: "+toString(armPot.GetAverageVoltage(),11)); //logMsg("Converted armPot value: "+toString(armPot.GetAverageVoltage(),11)); } if (i % 100 == 0 && compressing && compressor.GetPressureSwitchValue() == 1) { compressor.Stop(); compressing = false; logMsg("Stopping the compressor",2); } if (i % 100 == 0 && !compressing && compressor.GetPressureSwitchValue() == 0) { compressor.Start(); compressing = true; logMsg("Starting the compressor... again",2); } updateDashboard(); if (Lstick.GetRawButton(1)==1&&(upLimit>=potToDegrees(armPot.GetAverageVoltage()))) { //Move arm motors based on throttle if (collectorExtended == false) { shooting = false; logMsg("Collector is NOT extended, not going to fire",17); } if(collectorExtended == true) { shooting = true; logMsg("Firing",13); logMsg("Collector is extended, going to fire",17); shootRobot(throttle, false); setMotorValue(6, 1, 1); } } else if(Lstick.GetRawButton(1)==1&&(upLimit<=potToDegrees(armPot.GetAverageVoltage()))) { shooting = false; logMsg("Stopping shooter motor",13); logMsg("Stopping collector motor",17); shootRobot(0, true); } else if (Lstick.GetRawButton(2)==1) { //Reverse the arm motors shooting = false; if (collectorExtended == false) { logMsg("Collector is not extended, not going to fire",17); } if (collectorExtended == true) { shootRobot(-DownSpeed, false); logMsg("Collector is extended, going to fire",17); } } else { shooting = false; //Stop all motors shootRobot(0, true); } if (Rstick.GetRawButton(9)==1) { collectorExtended = true; collectorSole1.Set(false); collectorSole2.Set(true); } else if (Rstick.GetRawButton(10)==1) { collectorExtended = false; collectorSole1.Set(true); collectorSole2.Set(false); } if (Lstick.GetRawButton(11)==1) { setMotorValue(6, 1, 1); } else if (Lstick.GetRawButton(12)==1) { setMotorValue(6, 1, 255); } else if (!shooting) { setMotorValue(6, 1, 0); } cur++; i++; Wait(0.005f); } } }; START_ROBOT_CLASS(RobotDemo);