steak/include/okapi/impl/chassis/controller/chassisControllerBuilder.hpp
2024-09-18 13:05:17 -04:00

507 lines
20 KiB
C++

/*
* This Source Code Form is subject to the terms of the Mozilla Public
* License, v. 2.0. If a copy of the MPL was not distributed with this
* file, You can obtain one at http://mozilla.org/MPL/2.0/.
*/
#pragma once
#include "okapi/api/chassis/controller/chassisControllerIntegrated.hpp"
#include "okapi/api/chassis/controller/chassisControllerPid.hpp"
#include "okapi/api/chassis/controller/defaultOdomChassisController.hpp"
#include "okapi/api/chassis/model/hDriveModel.hpp"
#include "okapi/api/chassis/model/skidSteerModel.hpp"
#include "okapi/api/chassis/model/xDriveModel.hpp"
#include "okapi/api/util/logging.hpp"
#include "okapi/api/util/mathUtil.hpp"
#include "okapi/impl/device/motor/motor.hpp"
#include "okapi/impl/device/motor/motorGroup.hpp"
#include "okapi/impl/device/rotarysensor/adiEncoder.hpp"
#include "okapi/impl/device/rotarysensor/integratedEncoder.hpp"
#include "okapi/impl/device/rotarysensor/rotationSensor.hpp"
#include "okapi/impl/util/timeUtilFactory.hpp"
namespace okapi {
class ChassisControllerBuilder {
public:
/**
* A builder that creates ChassisControllers. Use this to create your ChassisController.
*
* @param ilogger The logger this instance will log to.
*/
explicit ChassisControllerBuilder(
const std::shared_ptr<Logger> &ilogger = Logger::getDefaultLogger());
/**
* Sets the motors using a skid-steer layout.
*
* @param ileft The left motor.
* @param iright The right motor.
* @return An ongoing builder.
*/
ChassisControllerBuilder &withMotors(const Motor &ileft, const Motor &iright);
/**
* Sets the motors using a skid-steer layout.
*
* @param ileft The left motor.
* @param iright The right motor.
* @return An ongoing builder.
*/
ChassisControllerBuilder &withMotors(const MotorGroup &ileft, const MotorGroup &iright);
/**
* Sets the motors using a skid-steer layout.
*
* @param ileft The left motor.
* @param iright The right motor.
* @return An ongoing builder.
*/
ChassisControllerBuilder &withMotors(const std::shared_ptr<AbstractMotor> &ileft,
const std::shared_ptr<AbstractMotor> &iright);
/**
* Sets the motors using an x-drive layout.
*
* @param itopLeft The top left motor.
* @param itopRight The top right motor.
* @param ibottomRight The bottom right motor.
* @param ibottomLeft The bottom left motor.
* @return An ongoing builder.
*/
ChassisControllerBuilder &withMotors(const Motor &itopLeft,
const Motor &itopRight,
const Motor &ibottomRight,
const Motor &ibottomLeft);
/**
* Sets the motors using an x-drive layout.
*
* @param itopLeft The top left motor.
* @param itopRight The top right motor.
* @param ibottomRight The bottom right motor.
* @param ibottomLeft The bottom left motor.
* @return An ongoing builder.
*/
ChassisControllerBuilder &withMotors(const MotorGroup &itopLeft,
const MotorGroup &itopRight,
const MotorGroup &ibottomRight,
const MotorGroup &ibottomLeft);
/**
* Sets the motors using an x-drive layout.
*
* @param itopLeft The top left motor.
* @param itopRight The top right motor.
* @param ibottomRight The bottom right motor.
* @param ibottomLeft The bottom left motor.
* @return An ongoing builder.
*/
ChassisControllerBuilder &withMotors(const std::shared_ptr<AbstractMotor> &itopLeft,
const std::shared_ptr<AbstractMotor> &itopRight,
const std::shared_ptr<AbstractMotor> &ibottomRight,
const std::shared_ptr<AbstractMotor> &ibottomLeft);
/**
* Sets the motors using an h-drive layout.
*
* @param ileft The left motor.
* @param iright The right motor.
* @param imiddle The middle motor.
* @return An ongoing builder.
*/
ChassisControllerBuilder &
withMotors(const Motor &ileft, const Motor &iright, const Motor &imiddle);
/**
* Sets the motors using an h-drive layout.
*
* @param ileft The left motor.
* @param iright The right motor.
* @param imiddle The middle motor.
* @return An ongoing builder.
*/
ChassisControllerBuilder &
withMotors(const MotorGroup &ileft, const MotorGroup &iright, const MotorGroup &imiddle);
/**
* Sets the motors using an h-drive layout.
*
* @param ileft The left motor.
* @param iright The right motor.
* @param imiddle The middle motor.
* @return An ongoing builder.
*/
ChassisControllerBuilder &
withMotors(const MotorGroup &ileft, const MotorGroup &iright, const Motor &imiddle);
/**
* Sets the motors using an h-drive layout.
*
* @param ileft The left motor.
* @param iright The right motor.
* @param imiddle The middle motor.
* @return An ongoing builder.
*/
ChassisControllerBuilder &withMotors(const std::shared_ptr<AbstractMotor> &ileft,
const std::shared_ptr<AbstractMotor> &iright,
const std::shared_ptr<AbstractMotor> &imiddle);
/**
* Sets the sensors. The default sensors are the motor's integrated encoders.
*
* @param ileft The left side sensor.
* @param iright The right side sensor.
* @return An ongoing builder.
*/
ChassisControllerBuilder &withSensors(const ADIEncoder &ileft, const ADIEncoder &iright);
/**
* Sets the sensors. The default sensors are the motor's integrated encoders.
*
* @param ileft The left side sensor.
* @param iright The right side sensor.
* @param imiddle The middle sensor.
* @return An ongoing builder.
*/
ChassisControllerBuilder &
withSensors(const ADIEncoder &ileft, const ADIEncoder &iright, const ADIEncoder &imiddle);
/**
* Sets the sensors. The default sensors are the motor's integrated encoders.
*
* @param ileft The left side sensor.
* @param iright The right side sensor.
* @return An ongoing builder.
*/
ChassisControllerBuilder &withSensors(const RotationSensor &ileft, const RotationSensor &iright);
/**
* Sets the sensors. The default sensors are the motor's integrated encoders.
*
* @param ileft The left side sensor.
* @param iright The right side sensor.
* @param imiddle The middle sensor.
* @return An ongoing builder.
*/
ChassisControllerBuilder &withSensors(const RotationSensor &ileft,
const RotationSensor &iright,
const RotationSensor &imiddle);
/**
* Sets the sensors. The default sensors are the motor's integrated encoders.
*
* @param ileft The left side sensor.
* @param iright The right side sensor.
* @return An ongoing builder.
*/
ChassisControllerBuilder &withSensors(const IntegratedEncoder &ileft,
const IntegratedEncoder &iright);
/**
* Sets the sensors. The default sensors are the motor's integrated encoders.
*
* @param ileft The left side sensor.
* @param iright The right side sensor.
* @param imiddle The middle sensor.
* @return An ongoing builder.
*/
ChassisControllerBuilder &withSensors(const IntegratedEncoder &ileft,
const IntegratedEncoder &iright,
const ADIEncoder &imiddle);
/**
* Sets the sensors. The default sensors are the motor's integrated encoders.
*
* @param ileft The left side sensor.
* @param iright The right side sensor.
* @return An ongoing builder.
*/
ChassisControllerBuilder &withSensors(const std::shared_ptr<ContinuousRotarySensor> &ileft,
const std::shared_ptr<ContinuousRotarySensor> &iright);
/**
* Sets the sensors. The default sensors are the motor's integrated encoders.
*
* @param ileft The left side sensor.
* @param iright The right side sensor.
* @param imiddle The middle sensor.
* @return An ongoing builder.
*/
ChassisControllerBuilder &withSensors(const std::shared_ptr<ContinuousRotarySensor> &ileft,
const std::shared_ptr<ContinuousRotarySensor> &iright,
const std::shared_ptr<ContinuousRotarySensor> &imiddle);
/**
* Sets the PID controller gains, causing the builder to generate a ChassisControllerPID. Uses the
* turn controller's gains for the angle controller's gains.
*
* @param idistanceGains The distance controller's gains.
* @param iturnGains The turn controller's gains.
* @return An ongoing builder.
*/
ChassisControllerBuilder &withGains(const IterativePosPIDController::Gains &idistanceGains,
const IterativePosPIDController::Gains &iturnGains);
/**
* Sets the PID controller gains, causing the builder to generate a ChassisControllerPID.
*
* @param idistanceGains The distance controller's gains.
* @param iturnGains The turn controller's gains.
* @param iangleGains The angle controller's gains.
* @return An ongoing builder.
*/
ChassisControllerBuilder &withGains(const IterativePosPIDController::Gains &idistanceGains,
const IterativePosPIDController::Gains &iturnGains,
const IterativePosPIDController::Gains &iangleGains);
/**
* Sets the odometry information, causing the builder to generate an Odometry variant.
*
* @param imode The new default StateMode used to interpret target points and query the Odometry
* state.
* @param imoveThreshold The minimum length movement.
* @param iturnThreshold The minimum angle turn.
* @return An ongoing builder.
*/
ChassisControllerBuilder &withOdometry(const StateMode &imode = StateMode::FRAME_TRANSFORMATION,
const QLength &imoveThreshold = 0_mm,
const QAngle &iturnThreshold = 0_deg);
/**
* Sets the odometry information, causing the builder to generate an Odometry variant.
*
* @param iodomScales The ChassisScales used just for odometry (if they are different than those
* for the drive).
* @param imode The new default StateMode used to interpret target points and query the Odometry
* state.
* @param imoveThreshold The minimum length movement.
* @param iturnThreshold The minimum angle turn.
* @return An ongoing builder.
*/
ChassisControllerBuilder &withOdometry(const ChassisScales &iodomScales,
const StateMode &imode = StateMode::FRAME_TRANSFORMATION,
const QLength &imoveThreshold = 0_mm,
const QAngle &iturnThreshold = 0_deg);
/**
* Sets the odometry information, causing the builder to generate an Odometry variant.
*
* @param iodometry The odometry object.
* @param imode The new default StateMode used to interpret target points and query the Odometry
* state.
* @param imoveThreshold The minimum length movement.
* @param iturnThreshold The minimum angle turn.
* @return An ongoing builder.
*/
ChassisControllerBuilder &withOdometry(std::shared_ptr<Odometry> iodometry,
const StateMode &imode = StateMode::FRAME_TRANSFORMATION,
const QLength &imoveThreshold = 0_mm,
const QAngle &iturnThreshold = 0_deg);
/**
* Sets the derivative filters. Uses a PassthroughFilter by default.
*
* @param idistanceFilter The distance controller's filter.
* @param iturnFilter The turn controller's filter.
* @param iangleFilter The angle controller's filter.
* @return An ongoing builder.
*/
ChassisControllerBuilder &withDerivativeFilters(
std::unique_ptr<Filter> idistanceFilter,
std::unique_ptr<Filter> iturnFilter = std::make_unique<PassthroughFilter>(),
std::unique_ptr<Filter> iangleFilter = std::make_unique<PassthroughFilter>());
/**
* Sets the chassis dimensions.
*
* @param igearset The gearset in the drive motors.
* @param iscales The ChassisScales for the base.
* @return An ongoing builder.
*/
ChassisControllerBuilder &withDimensions(const AbstractMotor::GearsetRatioPair &igearset,
const ChassisScales &iscales);
/**
* Sets the max velocity. Overrides the max velocity of the gearset.
*
* @param imaxVelocity The max velocity.
* @return An ongoing builder.
*/
ChassisControllerBuilder &withMaxVelocity(double imaxVelocity);
/**
* Sets the max voltage. The default is `12000`.
*
* @param imaxVoltage The max voltage.
* @return An ongoing builder.
*/
ChassisControllerBuilder &withMaxVoltage(double imaxVoltage);
/**
* Sets the TimeUtilFactory used when building a ChassisController. This instance will be given
* to the ChassisController (not to controllers it uses). The default is the static
* TimeUtilFactory.
*
* @param itimeUtilFactory The TimeUtilFactory.
* @return An ongoing builder.
*/
ChassisControllerBuilder &
withChassisControllerTimeUtilFactory(const TimeUtilFactory &itimeUtilFactory);
/**
* Sets the TimeUtilFactory used when building a ClosedLoopController. This instance will be given
* to any ClosedLoopController instances. The default is the static TimeUtilFactory.
*
* @param itimeUtilFactory The TimeUtilFactory.
* @return An ongoing builder.
*/
ChassisControllerBuilder &
withClosedLoopControllerTimeUtilFactory(const TimeUtilFactory &itimeUtilFactory);
/**
* Creates a new ConfigurableTimeUtilFactory with the given parameters. Given to any
* ClosedLoopController instances.
*
* @param iatTargetError The minimum error to be considered settled.
* @param iatTargetDerivative The minimum error derivative to be considered settled.
* @param iatTargetTime The minimum time within atTargetError to be considered settled.
* @return An ongoing builder.
*/
ChassisControllerBuilder &withClosedLoopControllerTimeUtil(double iatTargetError = 50,
double iatTargetDerivative = 5,
const QTime &iatTargetTime = 250_ms);
/**
* Sets the TimeUtilFactory used when building an Odometry. The default is the static
* TimeUtilFactory.
*
* @param itimeUtilFactory The TimeUtilFactory.
* @return An ongoing builder.
*/
ChassisControllerBuilder &withOdometryTimeUtilFactory(const TimeUtilFactory &itimeUtilFactory);
/**
* Sets the logger used for the ChassisController and ClosedLoopControllers.
*
* @param ilogger The logger.
* @return An ongoing builder.
*/
ChassisControllerBuilder &withLogger(const std::shared_ptr<Logger> &ilogger);
/**
* Parents the internal tasks started by this builder to the current task, meaning they will be
* deleted once the current task is deleted. The `initialize` and `competition_initialize` tasks
* are never parented to. This is the default behavior.
*
* Read more about this in the [builders and tasks tutorial]
* (docs/tutorials/concepts/builders-and-tasks.md).
*
* @return An ongoing builder.
*/
ChassisControllerBuilder &parentedToCurrentTask();
/**
* Prevents parenting the internal tasks started by this builder to the current task, meaning they
* will not be deleted once the current task is deleted. This can cause runaway tasks, but is
* sometimes the desired behavior (e.x., if you want to use this builder once in `autonomous` and
* then again in `opcontrol`).
*
* Read more about this in the [builders and tasks tutorial]
* (docs/tutorials/concepts/builders-and-tasks.md).
*
* @return An ongoing builder.
*/
ChassisControllerBuilder &notParentedToCurrentTask();
/**
* Builds the ChassisController. Throws a std::runtime_exception if no motors were set or if no
* dimensions were set.
*
* @return A fully built ChassisController.
*/
std::shared_ptr<ChassisController> build();
/**
* Builds the OdomChassisController. Throws a std::runtime_exception if no motors were set, if no
* dimensions were set, or if no odometry information was passed.
*
* @return A fully built OdomChassisController.
*/
std::shared_ptr<OdomChassisController> buildOdometry();
private:
std::shared_ptr<Logger> logger;
struct SkidSteerMotors {
std::shared_ptr<AbstractMotor> left;
std::shared_ptr<AbstractMotor> right;
};
struct XDriveMotors {
std::shared_ptr<AbstractMotor> topLeft;
std::shared_ptr<AbstractMotor> topRight;
std::shared_ptr<AbstractMotor> bottomRight;
std::shared_ptr<AbstractMotor> bottomLeft;
};
struct HDriveMotors {
std::shared_ptr<AbstractMotor> left;
std::shared_ptr<AbstractMotor> right;
std::shared_ptr<AbstractMotor> middle;
};
enum class DriveMode { SkidSteer, XDrive, HDrive };
bool hasMotors{false}; // Used to verify motors were passed
DriveMode driveMode{DriveMode::SkidSteer};
SkidSteerMotors skidSteerMotors;
XDriveMotors xDriveMotors;
HDriveMotors hDriveMotors;
bool sensorsSetByUser{false}; // Used so motors don't overwrite sensors set manually
std::shared_ptr<ContinuousRotarySensor> leftSensor{nullptr};
std::shared_ptr<ContinuousRotarySensor> rightSensor{nullptr};
std::shared_ptr<ContinuousRotarySensor> middleSensor{nullptr};
bool hasGains{false}; // Whether gains were passed, no gains means CCI
IterativePosPIDController::Gains distanceGains;
std::unique_ptr<Filter> distanceFilter = std::make_unique<PassthroughFilter>();
IterativePosPIDController::Gains angleGains;
std::unique_ptr<Filter> angleFilter = std::make_unique<PassthroughFilter>();
IterativePosPIDController::Gains turnGains;
std::unique_ptr<Filter> turnFilter = std::make_unique<PassthroughFilter>();
TimeUtilFactory chassisControllerTimeUtilFactory = TimeUtilFactory();
TimeUtilFactory closedLoopControllerTimeUtilFactory = TimeUtilFactory();
TimeUtilFactory odometryTimeUtilFactory = TimeUtilFactory();
AbstractMotor::GearsetRatioPair gearset{AbstractMotor::gearset::invalid, 1.0};
ChassisScales driveScales{{1, 1}, imev5GreenTPR};
bool differentOdomScales{false};
ChassisScales odomScales{{1, 1}, imev5GreenTPR};
std::shared_ptr<Logger> controllerLogger = Logger::getDefaultLogger();
bool hasOdom{false}; // Whether odometry was passed
std::shared_ptr<Odometry> odometry;
StateMode stateMode;
QLength moveThreshold;
QAngle turnThreshold;
bool maxVelSetByUser{false}; // Used so motors don't overwrite maxVelocity
double maxVelocity{600};
double maxVoltage{12000};
bool isParentedToCurrentTask{true};
std::shared_ptr<ChassisControllerPID> buildCCPID();
std::shared_ptr<ChassisControllerIntegrated> buildCCI();
std::shared_ptr<DefaultOdomChassisController>
buildDOCC(std::shared_ptr<ChassisController> chassisController);
std::shared_ptr<ChassisModel> makeChassisModel();
std::shared_ptr<SkidSteerModel> makeSkidSteerModel();
std::shared_ptr<XDriveModel> makeXDriveModel();
std::shared_ptr<HDriveModel> makeHDriveModel();
};
} // namespace okapi