Compare commits
17 Commits
v9.0
...
branch-car
Author | SHA1 | Date | |
---|---|---|---|
6d0eadecd1 | |||
b779a61a7c | |||
1a1ef3b2dc | |||
4ecd419511 | |||
9e8609e68d | |||
832de95fc1 | |||
63158215d0 | |||
6e6e626816 | |||
b5aca8b93c | |||
8a0aa126b6 | |||
6938e09b2b | |||
e86c31771e | |||
0baf961680 | |||
1da45a36c6 | |||
c303962469 | |||
f326c0d033 | |||
c023e97a6a |
@ -1,8 +1,8 @@
|
|||||||
<?xml version="1.0" encoding="utf-8"?>
|
<?xml version="1.0" encoding="utf-8"?>
|
||||||
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
|
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
|
||||||
xmlns:tools="http://schemas.android.com/tools"
|
xmlns:tools="http://schemas.android.com/tools"
|
||||||
android:versionCode="51"
|
android:versionCode="53"
|
||||||
android:versionName="9.0">
|
android:versionName="9.1">
|
||||||
|
|
||||||
<uses-permission android:name="android.permission.RECEIVE_BOOT_COMPLETED" />
|
<uses-permission android:name="android.permission.RECEIVE_BOOT_COMPLETED" />
|
||||||
|
|
||||||
|
@ -29,6 +29,7 @@
|
|||||||
|
|
||||||
package org.firstinspires.ftc.robotcontroller.external.samples;
|
package org.firstinspires.ftc.robotcontroller.external.samples;
|
||||||
|
|
||||||
|
import android.util.Size;
|
||||||
import com.qualcomm.robotcore.eventloop.opmode.Disabled;
|
import com.qualcomm.robotcore.eventloop.opmode.Disabled;
|
||||||
import com.qualcomm.robotcore.eventloop.opmode.LinearOpMode;
|
import com.qualcomm.robotcore.eventloop.opmode.LinearOpMode;
|
||||||
import com.qualcomm.robotcore.eventloop.opmode.TeleOp;
|
import com.qualcomm.robotcore.eventloop.opmode.TeleOp;
|
||||||
@ -44,6 +45,23 @@ import java.util.List;
|
|||||||
* This OpMode illustrates the basics of AprilTag recognition and pose estimation,
|
* This OpMode illustrates the basics of AprilTag recognition and pose estimation,
|
||||||
* including Java Builder structures for specifying Vision parameters.
|
* including Java Builder structures for specifying Vision parameters.
|
||||||
*
|
*
|
||||||
|
* For an introduction to AprilTags, see the FTC-DOCS link below:
|
||||||
|
* https://ftc-docs.firstinspires.org/en/latest/apriltag/vision_portal/apriltag_intro/apriltag-intro.html
|
||||||
|
*
|
||||||
|
* In this sample, any visible tag ID will be detected and displayed, but only tags that are included in the default
|
||||||
|
* "TagLibrary" will have their position and orientation information displayed. This default TagLibrary contains
|
||||||
|
* the current Season's AprilTags and a small set of "test Tags" in the high number range.
|
||||||
|
*
|
||||||
|
* When an AprilTag in the TagLibrary is detected, the SDK provides location and orientation of the tag, relative to the camera.
|
||||||
|
* This information is provided in the "ftcPose" member of the returned "detection", and is explained in the ftc-docs page linked below.
|
||||||
|
* https://ftc-docs.firstinspires.org/apriltag-detection-values
|
||||||
|
*
|
||||||
|
* To experiment with using AprilTags to navigate, try out these two driving samples:
|
||||||
|
* RobotAutoDriveToAprilTagOmni and RobotAutoDriveToAprilTagTank
|
||||||
|
*
|
||||||
|
* There are many "default" VisionPortal and AprilTag configuration parameters that may be overridden if desired.
|
||||||
|
* These default parameters are shown as comments in the code below.
|
||||||
|
*
|
||||||
* Use Android Studio to Copy this Class, and Paste it into your team's code folder with a new name.
|
* Use Android Studio to Copy this Class, and Paste it into your team's code folder with a new name.
|
||||||
* Remove or comment out the @Disabled line to add this OpMode to the Driver Station OpMode list.
|
* Remove or comment out the @Disabled line to add this OpMode to the Driver Station OpMode list.
|
||||||
*/
|
*/
|
||||||
@ -106,6 +124,8 @@ public class ConceptAprilTag extends LinearOpMode {
|
|||||||
|
|
||||||
// Create the AprilTag processor.
|
// Create the AprilTag processor.
|
||||||
aprilTag = new AprilTagProcessor.Builder()
|
aprilTag = new AprilTagProcessor.Builder()
|
||||||
|
|
||||||
|
// The following default settings are available to un-comment and edit as needed.
|
||||||
//.setDrawAxes(false)
|
//.setDrawAxes(false)
|
||||||
//.setDrawCubeProjection(false)
|
//.setDrawCubeProjection(false)
|
||||||
//.setDrawTagOutline(true)
|
//.setDrawTagOutline(true)
|
||||||
@ -117,11 +137,19 @@ public class ConceptAprilTag extends LinearOpMode {
|
|||||||
// If you do not manually specify calibration parameters, the SDK will attempt
|
// If you do not manually specify calibration parameters, the SDK will attempt
|
||||||
// to load a predefined calibration for your camera.
|
// to load a predefined calibration for your camera.
|
||||||
//.setLensIntrinsics(578.272, 578.272, 402.145, 221.506)
|
//.setLensIntrinsics(578.272, 578.272, 402.145, 221.506)
|
||||||
|
|
||||||
// ... these parameters are fx, fy, cx, cy.
|
// ... these parameters are fx, fy, cx, cy.
|
||||||
|
|
||||||
.build();
|
.build();
|
||||||
|
|
||||||
|
// Adjust Image Decimation to trade-off detection-range for detection-rate.
|
||||||
|
// eg: Some typical detection data using a Logitech C920 WebCam
|
||||||
|
// Decimation = 1 .. Detect 2" Tag from 10 feet away at 10 Frames per second
|
||||||
|
// Decimation = 2 .. Detect 2" Tag from 6 feet away at 22 Frames per second
|
||||||
|
// Decimation = 3 .. Detect 2" Tag from 4 feet away at 30 Frames Per Second (default)
|
||||||
|
// Decimation = 3 .. Detect 5" Tag from 10 feet away at 30 Frames Per Second (default)
|
||||||
|
// Note: Decimation can be changed on-the-fly to adapt during a match.
|
||||||
|
//aprilTag.setDecimation(3);
|
||||||
|
|
||||||
// Create the vision portal by using a builder.
|
// Create the vision portal by using a builder.
|
||||||
VisionPortal.Builder builder = new VisionPortal.Builder();
|
VisionPortal.Builder builder = new VisionPortal.Builder();
|
||||||
|
|
||||||
@ -136,7 +164,7 @@ public class ConceptAprilTag extends LinearOpMode {
|
|||||||
//builder.setCameraResolution(new Size(640, 480));
|
//builder.setCameraResolution(new Size(640, 480));
|
||||||
|
|
||||||
// Enable the RC preview (LiveView). Set "false" to omit camera monitoring.
|
// Enable the RC preview (LiveView). Set "false" to omit camera monitoring.
|
||||||
//builder.enableCameraMonitoring(true);
|
//builder.enableLiveView(true);
|
||||||
|
|
||||||
// Set the stream format; MJPEG uses less bandwidth than default YUY2.
|
// Set the stream format; MJPEG uses less bandwidth than default YUY2.
|
||||||
//builder.setStreamFormat(VisionPortal.StreamFormat.YUY2);
|
//builder.setStreamFormat(VisionPortal.StreamFormat.YUY2);
|
||||||
|
@ -44,6 +44,20 @@ import java.util.List;
|
|||||||
* This OpMode illustrates the basics of AprilTag recognition and pose estimation, using
|
* This OpMode illustrates the basics of AprilTag recognition and pose estimation, using
|
||||||
* the easy way.
|
* the easy way.
|
||||||
*
|
*
|
||||||
|
* For an introduction to AprilTags, see the FTC-DOCS link below:
|
||||||
|
* https://ftc-docs.firstinspires.org/en/latest/apriltag/vision_portal/apriltag_intro/apriltag-intro.html
|
||||||
|
*
|
||||||
|
* In this sample, any visible tag ID will be detected and displayed, but only tags that are included in the default
|
||||||
|
* "TagLibrary" will have their position and orientation information displayed. This default TagLibrary contains
|
||||||
|
* the current Season's AprilTags and a small set of "test Tags" in the high number range.
|
||||||
|
*
|
||||||
|
* When an AprilTag in the TagLibrary is detected, the SDK provides location and orientation of the tag, relative to the camera.
|
||||||
|
* This information is provided in the "ftcPose" member of the returned "detection", and is explained in the ftc-docs page linked below.
|
||||||
|
* https://ftc-docs.firstinspires.org/apriltag-detection-values
|
||||||
|
*
|
||||||
|
* To experiment with using AprilTags to navigate, try out these two driving samples:
|
||||||
|
* RobotAutoDriveToAprilTagOmni and RobotAutoDriveToAprilTagTank
|
||||||
|
*
|
||||||
* Use Android Studio to Copy this Class, and Paste it into your team's code folder with a new name.
|
* Use Android Studio to Copy this Class, and Paste it into your team's code folder with a new name.
|
||||||
* Remove or comment out the @Disabled line to add this OpMode to the Driver Station OpMode list.
|
* Remove or comment out the @Disabled line to add this OpMode to the Driver Station OpMode list.
|
||||||
*/
|
*/
|
||||||
|
@ -29,6 +29,7 @@
|
|||||||
|
|
||||||
package org.firstinspires.ftc.robotcontroller.external.samples;
|
package org.firstinspires.ftc.robotcontroller.external.samples;
|
||||||
|
|
||||||
|
import android.util.Size;
|
||||||
import com.qualcomm.robotcore.eventloop.opmode.Disabled;
|
import com.qualcomm.robotcore.eventloop.opmode.Disabled;
|
||||||
import com.qualcomm.robotcore.eventloop.opmode.LinearOpMode;
|
import com.qualcomm.robotcore.eventloop.opmode.LinearOpMode;
|
||||||
import com.qualcomm.robotcore.eventloop.opmode.TeleOp;
|
import com.qualcomm.robotcore.eventloop.opmode.TeleOp;
|
||||||
@ -53,6 +54,17 @@ public class ConceptTensorFlowObjectDetection extends LinearOpMode {
|
|||||||
|
|
||||||
private static final boolean USE_WEBCAM = true; // true for webcam, false for phone camera
|
private static final boolean USE_WEBCAM = true; // true for webcam, false for phone camera
|
||||||
|
|
||||||
|
// TFOD_MODEL_ASSET points to a model file stored in the project Asset location,
|
||||||
|
// this is only used for Android Studio when using models in Assets.
|
||||||
|
private static final String TFOD_MODEL_ASSET = "MyModelStoredAsAsset.tflite";
|
||||||
|
// TFOD_MODEL_FILE points to a model file stored onboard the Robot Controller's storage,
|
||||||
|
// this is used when uploading models directly to the RC using the model upload interface.
|
||||||
|
private static final String TFOD_MODEL_FILE = "/sdcard/FIRST/tflitemodels/myCustomModel.tflite";
|
||||||
|
// Define the labels recognized in the model for TFOD (must be in training order!)
|
||||||
|
private static final String[] LABELS = {
|
||||||
|
"Pixel",
|
||||||
|
};
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* The variable to store our instance of the TensorFlow Object Detection processor.
|
* The variable to store our instance of the TensorFlow Object Detection processor.
|
||||||
*/
|
*/
|
||||||
@ -107,11 +119,16 @@ public class ConceptTensorFlowObjectDetection extends LinearOpMode {
|
|||||||
// Create the TensorFlow processor by using a builder.
|
// Create the TensorFlow processor by using a builder.
|
||||||
tfod = new TfodProcessor.Builder()
|
tfod = new TfodProcessor.Builder()
|
||||||
|
|
||||||
// Use setModelAssetName() if the TF Model is built in as an asset.
|
// With the following lines commented out, the default TfodProcessor Builder
|
||||||
// Use setModelFileName() if you have downloaded a custom team model to the Robot Controller.
|
// will load the default model for the season. To define a custom model to load,
|
||||||
|
// choose one of the following:
|
||||||
|
// Use setModelAssetName() if the custom TF Model is built in as an asset (AS only).
|
||||||
|
// Use setModelFileName() if you have downloaded a custom team model to the Robot Controller.
|
||||||
//.setModelAssetName(TFOD_MODEL_ASSET)
|
//.setModelAssetName(TFOD_MODEL_ASSET)
|
||||||
//.setModelFileName(TFOD_MODEL_FILE)
|
//.setModelFileName(TFOD_MODEL_FILE)
|
||||||
|
|
||||||
|
// The following default settings are available to un-comment and edit as needed to
|
||||||
|
// set parameters for custom models.
|
||||||
//.setModelLabels(LABELS)
|
//.setModelLabels(LABELS)
|
||||||
//.setIsModelTensorFlow2(true)
|
//.setIsModelTensorFlow2(true)
|
||||||
//.setIsModelQuantized(true)
|
//.setIsModelQuantized(true)
|
||||||
@ -134,7 +151,7 @@ public class ConceptTensorFlowObjectDetection extends LinearOpMode {
|
|||||||
//builder.setCameraResolution(new Size(640, 480));
|
//builder.setCameraResolution(new Size(640, 480));
|
||||||
|
|
||||||
// Enable the RC preview (LiveView). Set "false" to omit camera monitoring.
|
// Enable the RC preview (LiveView). Set "false" to omit camera monitoring.
|
||||||
//builder.enableCameraMonitoring(true);
|
//builder.enableLiveView(true);
|
||||||
|
|
||||||
// Set the stream format; MJPEG uses less bandwidth than default YUY2.
|
// Set the stream format; MJPEG uses less bandwidth than default YUY2.
|
||||||
//builder.setStreamFormat(VisionPortal.StreamFormat.YUY2);
|
//builder.setStreamFormat(VisionPortal.StreamFormat.YUY2);
|
||||||
|
@ -49,6 +49,13 @@ import java.util.concurrent.TimeUnit;
|
|||||||
* This OpMode illustrates using a camera to locate and drive towards a specific AprilTag.
|
* This OpMode illustrates using a camera to locate and drive towards a specific AprilTag.
|
||||||
* The code assumes a Holonomic (Mecanum or X Drive) Robot.
|
* The code assumes a Holonomic (Mecanum or X Drive) Robot.
|
||||||
*
|
*
|
||||||
|
* For an introduction to AprilTags, see the ftc-docs link below:
|
||||||
|
* https://ftc-docs.firstinspires.org/en/latest/apriltag/vision_portal/apriltag_intro/apriltag-intro.html
|
||||||
|
*
|
||||||
|
* When an AprilTag in the TagLibrary is detected, the SDK provides location and orientation of the tag, relative to the camera.
|
||||||
|
* This information is provided in the "ftcPose" member of the returned "detection", and is explained in the ftc-docs page linked below.
|
||||||
|
* https://ftc-docs.firstinspires.org/apriltag-detection-values
|
||||||
|
*
|
||||||
* The drive goal is to rotate to keep the Tag centered in the camera, while strafing to be directly in front of the tag, and
|
* The drive goal is to rotate to keep the Tag centered in the camera, while strafing to be directly in front of the tag, and
|
||||||
* driving towards the tag to achieve the desired distance.
|
* driving towards the tag to achieve the desired distance.
|
||||||
* To reduce any motion blur (which will interrupt the detection process) the Camera exposure is reduced to a very low value (5mS)
|
* To reduce any motion blur (which will interrupt the detection process) the Camera exposure is reduced to a very low value (5mS)
|
||||||
@ -102,7 +109,7 @@ public class RobotAutoDriveToAprilTagOmni extends LinearOpMode
|
|||||||
private DcMotor rightBackDrive = null; // Used to control the right back drive wheel
|
private DcMotor rightBackDrive = null; // Used to control the right back drive wheel
|
||||||
|
|
||||||
private static final boolean USE_WEBCAM = true; // Set true to use a webcam, or false for a phone camera
|
private static final boolean USE_WEBCAM = true; // Set true to use a webcam, or false for a phone camera
|
||||||
private static final int DESIRED_TAG_ID = 0; // Choose the tag you want to approach or set to -1 for ANY tag.
|
private static final int DESIRED_TAG_ID = -1; // Choose the tag you want to approach or set to -1 for ANY tag.
|
||||||
private VisionPortal visionPortal; // Used to manage the video source.
|
private VisionPortal visionPortal; // Used to manage the video source.
|
||||||
private AprilTagProcessor aprilTag; // Used for managing the AprilTag detection process.
|
private AprilTagProcessor aprilTag; // Used for managing the AprilTag detection process.
|
||||||
private AprilTagDetection desiredTag = null; // Used to hold the data for a detected AprilTag
|
private AprilTagDetection desiredTag = null; // Used to hold the data for a detected AprilTag
|
||||||
@ -150,25 +157,33 @@ public class RobotAutoDriveToAprilTagOmni extends LinearOpMode
|
|||||||
// Step through the list of detected tags and look for a matching tag
|
// Step through the list of detected tags and look for a matching tag
|
||||||
List<AprilTagDetection> currentDetections = aprilTag.getDetections();
|
List<AprilTagDetection> currentDetections = aprilTag.getDetections();
|
||||||
for (AprilTagDetection detection : currentDetections) {
|
for (AprilTagDetection detection : currentDetections) {
|
||||||
if ((detection.metadata != null) &&
|
// Look to see if we have size info on this tag.
|
||||||
((DESIRED_TAG_ID < 0) || (detection.id == DESIRED_TAG_ID)) ){
|
if (detection.metadata != null) {
|
||||||
targetFound = true;
|
// Check to see if we want to track towards this tag.
|
||||||
desiredTag = detection;
|
if ((DESIRED_TAG_ID < 0) || (detection.id == DESIRED_TAG_ID)) {
|
||||||
break; // don't look any further.
|
// Yes, we want to use this tag.
|
||||||
|
targetFound = true;
|
||||||
|
desiredTag = detection;
|
||||||
|
break; // don't look any further.
|
||||||
|
} else {
|
||||||
|
// This tag is in the library, but we do not want to track it right now.
|
||||||
|
telemetry.addData("Skipping", "Tag ID %d is not desired", detection.id);
|
||||||
|
}
|
||||||
} else {
|
} else {
|
||||||
telemetry.addData("Unknown Target", "Tag ID %d is not in TagLibrary\n", detection.id);
|
// This tag is NOT in the library, so we don't have enough information to track to it.
|
||||||
|
telemetry.addData("Unknown", "Tag ID %d is not in TagLibrary", detection.id);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Tell the driver what we see, and what to do.
|
// Tell the driver what we see, and what to do.
|
||||||
if (targetFound) {
|
if (targetFound) {
|
||||||
telemetry.addData(">","HOLD Left-Bumper to Drive to Target\n");
|
telemetry.addData("\n>","HOLD Left-Bumper to Drive to Target\n");
|
||||||
telemetry.addData("Target", "ID %d (%s)", desiredTag.id, desiredTag.metadata.name);
|
telemetry.addData("Found", "ID %d (%s)", desiredTag.id, desiredTag.metadata.name);
|
||||||
telemetry.addData("Range", "%5.1f inches", desiredTag.ftcPose.range);
|
telemetry.addData("Range", "%5.1f inches", desiredTag.ftcPose.range);
|
||||||
telemetry.addData("Bearing","%3.0f degrees", desiredTag.ftcPose.bearing);
|
telemetry.addData("Bearing","%3.0f degrees", desiredTag.ftcPose.bearing);
|
||||||
telemetry.addData("Yaw","%3.0f degrees", desiredTag.ftcPose.yaw);
|
telemetry.addData("Yaw","%3.0f degrees", desiredTag.ftcPose.yaw);
|
||||||
} else {
|
} else {
|
||||||
telemetry.addData(">","Drive using joysticks to find valid target\n");
|
telemetry.addData("\n>","Drive using joysticks to find valid target\n");
|
||||||
}
|
}
|
||||||
|
|
||||||
// If Left Bumper is being pressed, AND we have found the desired target, Drive to target Automatically .
|
// If Left Bumper is being pressed, AND we have found the desired target, Drive to target Automatically .
|
||||||
@ -243,6 +258,15 @@ public class RobotAutoDriveToAprilTagOmni extends LinearOpMode
|
|||||||
// Create the AprilTag processor by using a builder.
|
// Create the AprilTag processor by using a builder.
|
||||||
aprilTag = new AprilTagProcessor.Builder().build();
|
aprilTag = new AprilTagProcessor.Builder().build();
|
||||||
|
|
||||||
|
// Adjust Image Decimation to trade-off detection-range for detection-rate.
|
||||||
|
// eg: Some typical detection data using a Logitech C920 WebCam
|
||||||
|
// Decimation = 1 .. Detect 2" Tag from 10 feet away at 10 Frames per second
|
||||||
|
// Decimation = 2 .. Detect 2" Tag from 6 feet away at 22 Frames per second
|
||||||
|
// Decimation = 3 .. Detect 2" Tag from 4 feet away at 30 Frames Per Second
|
||||||
|
// Decimation = 3 .. Detect 5" Tag from 10 feet away at 30 Frames Per Second
|
||||||
|
// Note: Decimation can be changed on-the-fly to adapt during a match.
|
||||||
|
aprilTag.setDecimation(2);
|
||||||
|
|
||||||
// Create the vision portal by using a builder.
|
// Create the vision portal by using a builder.
|
||||||
if (USE_WEBCAM) {
|
if (USE_WEBCAM) {
|
||||||
visionPortal = new VisionPortal.Builder()
|
visionPortal = new VisionPortal.Builder()
|
||||||
|
@ -49,6 +49,13 @@ import java.util.concurrent.TimeUnit;
|
|||||||
* This OpMode illustrates using a camera to locate and drive towards a specific AprilTag.
|
* This OpMode illustrates using a camera to locate and drive towards a specific AprilTag.
|
||||||
* The code assumes a basic two-wheel (Tank) Robot Drivetrain
|
* The code assumes a basic two-wheel (Tank) Robot Drivetrain
|
||||||
*
|
*
|
||||||
|
* For an introduction to AprilTags, see the ftc-docs link below:
|
||||||
|
* https://ftc-docs.firstinspires.org/en/latest/apriltag/vision_portal/apriltag_intro/apriltag-intro.html
|
||||||
|
*
|
||||||
|
* When an AprilTag in the TagLibrary is detected, the SDK provides location and orientation of the tag, relative to the camera.
|
||||||
|
* This information is provided in the "ftcPose" member of the returned "detection", and is explained in the ftc-docs page linked below.
|
||||||
|
* https://ftc-docs.firstinspires.org/apriltag-detection-values
|
||||||
|
*
|
||||||
* The driving goal is to rotate to keep the tag centered in the camera, while driving towards the tag to achieve the desired distance.
|
* The driving goal is to rotate to keep the tag centered in the camera, while driving towards the tag to achieve the desired distance.
|
||||||
* To reduce any motion blur (which will interrupt the detection process) the Camera exposure is reduced to a very low value (5mS)
|
* To reduce any motion blur (which will interrupt the detection process) the Camera exposure is reduced to a very low value (5mS)
|
||||||
* You can determine the best exposure and gain values by using the ConceptAprilTagOptimizeExposure OpMode in this Samples folder.
|
* You can determine the best exposure and gain values by using the ConceptAprilTagOptimizeExposure OpMode in this Samples folder.
|
||||||
@ -97,7 +104,7 @@ public class RobotAutoDriveToAprilTagTank extends LinearOpMode
|
|||||||
private DcMotor rightDrive = null; // Used to control the right drive wheel
|
private DcMotor rightDrive = null; // Used to control the right drive wheel
|
||||||
|
|
||||||
private static final boolean USE_WEBCAM = true; // Set true to use a webcam, or false for a phone camera
|
private static final boolean USE_WEBCAM = true; // Set true to use a webcam, or false for a phone camera
|
||||||
private static final int DESIRED_TAG_ID = 0; // Choose the tag you want to approach or set to -1 for ANY tag.
|
private static final int DESIRED_TAG_ID = -1; // Choose the tag you want to approach or set to -1 for ANY tag.
|
||||||
private VisionPortal visionPortal; // Used to manage the video source.
|
private VisionPortal visionPortal; // Used to manage the video source.
|
||||||
private AprilTagProcessor aprilTag; // Used for managing the AprilTag detection process.
|
private AprilTagProcessor aprilTag; // Used for managing the AprilTag detection process.
|
||||||
private AprilTagDetection desiredTag = null; // Used to hold the data for a detected AprilTag
|
private AprilTagDetection desiredTag = null; // Used to hold the data for a detected AprilTag
|
||||||
@ -140,24 +147,32 @@ public class RobotAutoDriveToAprilTagTank extends LinearOpMode
|
|||||||
// Step through the list of detected tags and look for a matching tag
|
// Step through the list of detected tags and look for a matching tag
|
||||||
List<AprilTagDetection> currentDetections = aprilTag.getDetections();
|
List<AprilTagDetection> currentDetections = aprilTag.getDetections();
|
||||||
for (AprilTagDetection detection : currentDetections) {
|
for (AprilTagDetection detection : currentDetections) {
|
||||||
if ((detection.metadata != null) &&
|
// Look to see if we have size info on this tag.
|
||||||
((DESIRED_TAG_ID < 0) || (detection.id == DESIRED_TAG_ID)) ){
|
if (detection.metadata != null) {
|
||||||
targetFound = true;
|
// Check to see if we want to track towards this tag.
|
||||||
desiredTag = detection;
|
if ((DESIRED_TAG_ID < 0) || (detection.id == DESIRED_TAG_ID)) {
|
||||||
break; // don't look any further.
|
// Yes, we want to use this tag.
|
||||||
|
targetFound = true;
|
||||||
|
desiredTag = detection;
|
||||||
|
break; // don't look any further.
|
||||||
|
} else {
|
||||||
|
// This tag is in the library, but we do not want to track it right now.
|
||||||
|
telemetry.addData("Skipping", "Tag ID %d is not desired", detection.id);
|
||||||
|
}
|
||||||
} else {
|
} else {
|
||||||
telemetry.addData("Unknown Target", "Tag ID %d is not in TagLibrary\n", detection.id);
|
// This tag is NOT in the library, so we don't have enough information to track to it.
|
||||||
|
telemetry.addData("Unknown", "Tag ID %d is not in TagLibrary", detection.id);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Tell the driver what we see, and what to do.
|
// Tell the driver what we see, and what to do.
|
||||||
if (targetFound) {
|
if (targetFound) {
|
||||||
telemetry.addData(">","HOLD Left-Bumper to Drive to Target\n");
|
telemetry.addData("\n>","HOLD Left-Bumper to Drive to Target\n");
|
||||||
telemetry.addData("Target", "ID %d (%s)", desiredTag.id, desiredTag.metadata.name);
|
telemetry.addData("Found", "ID %d (%s)", desiredTag.id, desiredTag.metadata.name);
|
||||||
telemetry.addData("Range", "%5.1f inches", desiredTag.ftcPose.range);
|
telemetry.addData("Range", "%5.1f inches", desiredTag.ftcPose.range);
|
||||||
telemetry.addData("Bearing","%3.0f degrees", desiredTag.ftcPose.bearing);
|
telemetry.addData("Bearing","%3.0f degrees", desiredTag.ftcPose.bearing);
|
||||||
} else {
|
} else {
|
||||||
telemetry.addData(">","Drive using joysticks to find valid target\n");
|
telemetry.addData("\n>","Drive using joysticks to find valid target\n");
|
||||||
}
|
}
|
||||||
|
|
||||||
// If Left Bumper is being pressed, AND we have found the desired target, Drive to target Automatically .
|
// If Left Bumper is being pressed, AND we have found the desired target, Drive to target Automatically .
|
||||||
@ -218,6 +233,15 @@ public class RobotAutoDriveToAprilTagTank extends LinearOpMode
|
|||||||
// Create the AprilTag processor by using a builder.
|
// Create the AprilTag processor by using a builder.
|
||||||
aprilTag = new AprilTagProcessor.Builder().build();
|
aprilTag = new AprilTagProcessor.Builder().build();
|
||||||
|
|
||||||
|
// Adjust Image Decimation to trade-off detection-range for detection-rate.
|
||||||
|
// eg: Some typical detection data using a Logitech C920 WebCam
|
||||||
|
// Decimation = 1 .. Detect 2" Tag from 10 feet away at 10 Frames per second
|
||||||
|
// Decimation = 2 .. Detect 2" Tag from 6 feet away at 22 Frames per second
|
||||||
|
// Decimation = 3 .. Detect 2" Tag from 4 feet away at 30 Frames Per Second
|
||||||
|
// Decimation = 3 .. Detect 5" Tag from 10 feet away at 30 Frames Per Second
|
||||||
|
// Note: Decimation can be changed on-the-fly to adapt during a match.
|
||||||
|
aprilTag.setDecimation(2);
|
||||||
|
|
||||||
// Create the vision portal by using a builder.
|
// Create the vision portal by using a builder.
|
||||||
if (USE_WEBCAM) {
|
if (USE_WEBCAM) {
|
||||||
visionPortal = new VisionPortal.Builder()
|
visionPortal = new VisionPortal.Builder()
|
||||||
|
@ -0,0 +1,78 @@
|
|||||||
|
/* Copyright (c) 2024 FIRST. All rights reserved.
|
||||||
|
*
|
||||||
|
* Redistribution and use in source and binary forms, with or without modification,
|
||||||
|
* are permitted (subject to the limitations in the disclaimer below) provided that
|
||||||
|
* the following conditions are met:
|
||||||
|
*
|
||||||
|
* Redistributions of source code must retain the above copyright notice, this list
|
||||||
|
* of conditions and the following disclaimer.
|
||||||
|
*
|
||||||
|
* Redistributions in binary form must reproduce the above copyright notice, this
|
||||||
|
* list of conditions and the following disclaimer in the documentation and/or
|
||||||
|
* other materials provided with the distribution.
|
||||||
|
*
|
||||||
|
* Neither the name of FIRST nor the names of its contributors may be used to endorse or
|
||||||
|
* promote products derived from this software without specific prior written permission.
|
||||||
|
*
|
||||||
|
* NO EXPRESS OR IMPLIED LICENSES TO ANY PARTY'S PATENT RIGHTS ARE GRANTED BY THIS
|
||||||
|
* LICENSE. THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
|
||||||
|
* "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO,
|
||||||
|
* THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
|
||||||
|
* ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE
|
||||||
|
* FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
|
||||||
|
* DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
|
||||||
|
* SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER
|
||||||
|
* CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY,
|
||||||
|
* OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
|
||||||
|
* OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
||||||
|
*/
|
||||||
|
|
||||||
|
package org.firstinspires.ftc.robotcontroller.external.samples;
|
||||||
|
|
||||||
|
import com.qualcomm.robotcore.eventloop.opmode.Disabled;
|
||||||
|
import com.qualcomm.robotcore.eventloop.opmode.LinearOpMode;
|
||||||
|
import com.qualcomm.robotcore.eventloop.opmode.TeleOp;
|
||||||
|
import com.qualcomm.robotcore.hardware.DigitalChannel;
|
||||||
|
|
||||||
|
/*
|
||||||
|
* This OpMode demonstrates how to use a digital channel.
|
||||||
|
*
|
||||||
|
* The OpMode assumes that the digital channel is configured with a name of "digitalTouch".
|
||||||
|
*
|
||||||
|
* Use Android Studio to Copy this Class, and Paste it into your team's code folder with a new name.
|
||||||
|
* Remove or comment out the @Disabled line to add this OpMode to the Driver Station OpMode list.
|
||||||
|
*/
|
||||||
|
@TeleOp(name = "Sensor: digital channel", group = "Sensor")
|
||||||
|
@Disabled
|
||||||
|
public class SensorDigitalTouch extends LinearOpMode {
|
||||||
|
DigitalChannel digitalTouch; // Digital channel Object
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void runOpMode() {
|
||||||
|
|
||||||
|
// get a reference to our touchSensor object.
|
||||||
|
digitalTouch = hardwareMap.get(DigitalChannel.class, "digitalTouch");
|
||||||
|
|
||||||
|
digitalTouch.setMode(DigitalChannel.Mode.INPUT);
|
||||||
|
telemetry.addData("DigitalTouchSensorExample", "Press start to continue...");
|
||||||
|
telemetry.update();
|
||||||
|
|
||||||
|
// wait for the start button to be pressed.
|
||||||
|
waitForStart();
|
||||||
|
|
||||||
|
// while the OpMode is active, loop and read the digital channel.
|
||||||
|
// Note we use opModeIsActive() as our loop condition because it is an interruptible method.
|
||||||
|
while (opModeIsActive()) {
|
||||||
|
|
||||||
|
// button is pressed if value returned is LOW or false.
|
||||||
|
// send the info back to driver station using telemetry function.
|
||||||
|
if (digitalTouch.getState() == false) {
|
||||||
|
telemetry.addData("Button", "PRESSED");
|
||||||
|
} else {
|
||||||
|
telemetry.addData("Button", "NOT PRESSED");
|
||||||
|
}
|
||||||
|
|
||||||
|
telemetry.update();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
10
IMU.md
Normal file
10
IMU.md
Normal file
@ -0,0 +1,10 @@
|
|||||||
|
# IMU
|
||||||
|
|
||||||
|
## Current list of IMUs
|
||||||
|
|
||||||
|
| Robot Name | Control Hub Firmware | Control Hub IMU | Expansion Hub? | Expansion Hub Firmware | Expansion Hub IMU |
|
||||||
|
|--------------------|----------------------|-----------------|-----------------|------------------------|-------------------|
|
||||||
|
| 14493-RC | 1.8.2 | None | Yes | 1.8.2 | None |
|
||||||
|
| 14493-b-RC | 1.8.2 | None | Yes | 1.8.2 | None |
|
||||||
|
| 14493-RC-Chassis | 1.8.2 | BNO055 | No | N/A | N/A |
|
||||||
|
|
52
README.md
52
README.md
@ -59,6 +59,58 @@ The readme.md file located in the [/TeamCode/src/main/java/org/firstinspires/ftc
|
|||||||
|
|
||||||
# Release Information
|
# Release Information
|
||||||
|
|
||||||
|
## Version 9.1 (20240215-115542)
|
||||||
|
|
||||||
|
### Enhancements
|
||||||
|
* Fixes a problem with Blocks: if the user closes a Block's warning balloon, it will still be closed next time the project is opened in the Blocks editor.
|
||||||
|
* In the Blocks editor, an alert concerning missing hardware devices is not shown if all the Blocks that use the missing hardware devices are disabled.
|
||||||
|
* Adds Blocks to support comparing property values CRServo.Direction, DCMotor.Direction, DCMotor.Mode, DCMotor.ZeroPowerBehavior, DigitalChannel.Mode, GyroSensor.HeadingMode, IrSeekerSensor.Mode, and Servo.Direction, to the corresponding enum Block.
|
||||||
|
* Improves OnBotJava auto-import to correctly import classes when used in certain situations.
|
||||||
|
* Improves OnBotJava autocomplete to provide better completion options in most cases.
|
||||||
|
* This fixes an issue where autocomplete would fail if a method with two or more formal parameters was defined.
|
||||||
|
* In OnBotJava, code folding support was added to expand and collapse code sections
|
||||||
|
* In OnBotJava, the copyright header is now automatically collapsed loading new files
|
||||||
|
* For all Blocks OpMode samples, intro comments have been moved to the RunOpMode comment balloon.
|
||||||
|
* The Clean up Blocks command in the Blocks editor now positions function Blocks so their comment balloons don't overlap other function Blocks.
|
||||||
|
* Added Blocks OpMode sample SensorTouch.
|
||||||
|
* Added Java OpMode sample SensorDigitalTouch.
|
||||||
|
* Several improvements to VisionPortal
|
||||||
|
* Adds option to control whether the stream is automatically started following a `.build()` call on a VisionPortal Builder
|
||||||
|
* Adds option to control whether the vision processing statistics overlay is rendered or not
|
||||||
|
* VisionPortals now implement the `CameraStreamSource` interface, allowing multiportal users to select which portal is routed to the DS in INIT by calling CameraStreamServer.getInstance().setSource(visionPortal). Can be selected via gamepad, between Camera Stream sessions.
|
||||||
|
* Add option to `AprilTagProcessor` to suppress calibration warnings
|
||||||
|
* Improves camera calibration warnings
|
||||||
|
* If a calibration is scaled, the resolution it was scaled from will be listed
|
||||||
|
* If calibrations exist with the wrong aspect ratio, the calibrated resolutions will be listed
|
||||||
|
* Fixes race condition which caused app crash when calling `stopStreaming()` immediately followed by `close()` on a VisionPortal
|
||||||
|
* Fixes IllegalStateException when calling `stopStreaming()` immediately after building a VisionPortal
|
||||||
|
* Added FTC Blocks counterparts to new Java methods:
|
||||||
|
* VisionPortal.Builder.setAutoStartStreamOnBuild
|
||||||
|
* VisionPortal.Builder.setShowStatsOverlay
|
||||||
|
* AprilTagProcessor.Builder.setSuppressCalibrationWarnings
|
||||||
|
* CameraStreamServer.setSource
|
||||||
|
|
||||||
|
### Bug Fixes
|
||||||
|
* Fixes a problem where OnBotJava does not apply font size settings to the editor.
|
||||||
|
* Updates EasyOpenCV dependency to v1.7.1
|
||||||
|
* Fixes inability to use EasyOpenCV CameraFactory in OnBotJava
|
||||||
|
* Fixes entire RC app crash when user pipeline throws an exception
|
||||||
|
* Fixes entire RC app crash when user user canvas annotator throws an exception
|
||||||
|
* Use the modern stacktrace display when handling user exceptions instead of the legacy ESTOP telemetry message
|
||||||
|
|
||||||
|
## Version 9.0.1 (20230929-083754)
|
||||||
|
|
||||||
|
### Enhancements
|
||||||
|
* Updates AprilTag samples to include Decimation and additional Comments. Also corrects misleading tag ID warnings
|
||||||
|
* Increases maximum size of Blocks inline comments to 140 characters
|
||||||
|
* Adds Blocks sample BasicOmniOpMode.
|
||||||
|
* Updated CENTERSTAGE library AprilTag orientation quaternions
|
||||||
|
* Thanks [@FromenActual](https://github.com/FromenActual)
|
||||||
|
* Updated Java Sample ConceptTensorFlowObjectDetection.java to include missing elements needed for custom model support.
|
||||||
|
|
||||||
|
### Bug Fixes
|
||||||
|
* Fixes a problem where after October 1 the Driver Station will report as obsolete on v9.0 and prompt the user to update.
|
||||||
|
|
||||||
## Version 9.0 (20230830-154348)
|
## Version 9.0 (20230830-154348)
|
||||||
|
|
||||||
### Breaking Changes
|
### Breaking Changes
|
||||||
|
@ -23,7 +23,17 @@ android {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
repositories {
|
||||||
|
maven {
|
||||||
|
url = 'https://maven.brott.dev/'
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
dependencies {
|
dependencies {
|
||||||
implementation project(':FtcRobotController')
|
implementation project(':FtcRobotController')
|
||||||
annotationProcessor files('lib/OpModeAnnotationProcessor.jar')
|
annotationProcessor files('lib/OpModeAnnotationProcessor.jar')
|
||||||
|
implementation "com.acmerobotics.roadrunner:ftc:0.1.12"
|
||||||
|
implementation "com.acmerobotics.roadrunner:core:1.0.0-beta8"
|
||||||
|
implementation "com.acmerobotics.roadrunner:actions:1.0.0-beta8"
|
||||||
|
implementation "com.acmerobotics.dashboard:dashboard:0.4.14"
|
||||||
}
|
}
|
||||||
|
@ -0,0 +1,78 @@
|
|||||||
|
package org.firstinspires.ftc.teamcode;
|
||||||
|
|
||||||
|
import android.util.Size;
|
||||||
|
|
||||||
|
import com.qualcomm.robotcore.eventloop.opmode.LinearOpMode;
|
||||||
|
import com.qualcomm.robotcore.eventloop.opmode.TeleOp;
|
||||||
|
|
||||||
|
import org.firstinspires.ftc.robotcore.external.hardware.camera.BuiltinCameraDirection;
|
||||||
|
import org.firstinspires.ftc.robotcore.external.hardware.camera.WebcamName;
|
||||||
|
import org.firstinspires.ftc.vision.VisionPortal;
|
||||||
|
|
||||||
|
import java.util.Locale;
|
||||||
|
|
||||||
|
@TeleOp(name = "CometBots Camera Demo (HDMI)", group = "[14493] CometBots")
|
||||||
|
public class CometBotsCameraDemo extends LinearOpMode {
|
||||||
|
|
||||||
|
final boolean USING_WEBCAM = true;
|
||||||
|
final BuiltinCameraDirection INTERNAL_CAM_DIR = BuiltinCameraDirection.FRONT;
|
||||||
|
final int RESOLUTION_WIDTH = 640;
|
||||||
|
final int RESOLUTION_HEIGHT = 480;
|
||||||
|
|
||||||
|
// Internal state
|
||||||
|
boolean lastX;
|
||||||
|
int frameCount;
|
||||||
|
long capReqTime;
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void runOpMode() throws InterruptedException {
|
||||||
|
|
||||||
|
telemetry.speak("Development - Camera Test");
|
||||||
|
|
||||||
|
VisionPortal portal;
|
||||||
|
|
||||||
|
if (USING_WEBCAM)
|
||||||
|
{
|
||||||
|
portal = new VisionPortal.Builder()
|
||||||
|
.setCamera(hardwareMap.get(WebcamName.class, "watch out"))
|
||||||
|
.setCameraResolution(new Size(RESOLUTION_WIDTH, RESOLUTION_HEIGHT))
|
||||||
|
.build();
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
portal = new VisionPortal.Builder()
|
||||||
|
.setCamera(INTERNAL_CAM_DIR)
|
||||||
|
.setCameraResolution(new Size(RESOLUTION_WIDTH, RESOLUTION_HEIGHT))
|
||||||
|
.build();
|
||||||
|
}
|
||||||
|
|
||||||
|
while (!isStopRequested())
|
||||||
|
{
|
||||||
|
boolean x = gamepad1.x;
|
||||||
|
|
||||||
|
if (x && !lastX)
|
||||||
|
{
|
||||||
|
portal.saveNextFrameRaw(String.format(Locale.US, "CameraFrameCapture-%06d", frameCount++));
|
||||||
|
capReqTime = System.currentTimeMillis();
|
||||||
|
}
|
||||||
|
|
||||||
|
lastX = x;
|
||||||
|
|
||||||
|
telemetry.addLine("######## Camera Capture Utility ########");
|
||||||
|
telemetry.addLine(String.format(Locale.US, " > Resolution: %dx%d", RESOLUTION_WIDTH, RESOLUTION_HEIGHT));
|
||||||
|
telemetry.addLine(" > Press X (or Square) to capture a frame");
|
||||||
|
telemetry.addData(" > Camera Status", portal.getCameraState());
|
||||||
|
|
||||||
|
if (capReqTime != 0)
|
||||||
|
{
|
||||||
|
telemetry.addLine("\nCaptured Frame!");
|
||||||
|
}
|
||||||
|
|
||||||
|
if (capReqTime != 0 && System.currentTimeMillis() - capReqTime > 1000)
|
||||||
|
{
|
||||||
|
capReqTime = 0;
|
||||||
|
}
|
||||||
|
telemetry.update();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
@ -0,0 +1,72 @@
|
|||||||
|
package org.firstinspires.ftc.teamcode;
|
||||||
|
|
||||||
|
import com.qualcomm.hardware.rev.RevHubOrientationOnRobot;
|
||||||
|
import com.qualcomm.robotcore.eventloop.opmode.LinearOpMode;
|
||||||
|
import com.qualcomm.robotcore.eventloop.opmode.TeleOp;
|
||||||
|
import com.qualcomm.robotcore.hardware.IMU;
|
||||||
|
|
||||||
|
import org.firstinspires.ftc.robotcore.external.navigation.AngleUnit;
|
||||||
|
import org.firstinspires.ftc.robotcore.external.navigation.AngularVelocity;
|
||||||
|
import org.firstinspires.ftc.robotcore.external.navigation.YawPitchRollAngles;
|
||||||
|
|
||||||
|
@TeleOp(name = "CometBots IMU Demo", group = "[14493] CometBots")
|
||||||
|
public class CometBotsIMUDemo extends LinearOpMode {
|
||||||
|
|
||||||
|
IMU imu;
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void runOpMode() throws InterruptedException {
|
||||||
|
|
||||||
|
// Retrieve and initialize the IMU.
|
||||||
|
// This sample expects the IMU to be in a REV Hub and named "imu".
|
||||||
|
imu = hardwareMap.get(IMU.class, "imu");
|
||||||
|
|
||||||
|
/* Define how the hub is mounted on the robot to get the correct Yaw, Pitch and Roll values.
|
||||||
|
*
|
||||||
|
* Two input parameters are required to fully specify the Orientation.
|
||||||
|
* The first parameter specifies the direction the printed logo on the Hub is pointing.
|
||||||
|
* The second parameter specifies the direction the USB connector on the Hub is pointing.
|
||||||
|
* All directions are relative to the robot, and left/right is as-viewed from behind the robot.
|
||||||
|
*/
|
||||||
|
|
||||||
|
/* The next two lines define Hub orientation.
|
||||||
|
* The Default Orientation (shown) is when a hub is mounted horizontally with the printed logo pointing UP and the USB port pointing FORWARD.
|
||||||
|
*
|
||||||
|
* To Do: EDIT these two lines to match YOUR mounting configuration.
|
||||||
|
*/
|
||||||
|
RevHubOrientationOnRobot.LogoFacingDirection logoDirection = RevHubOrientationOnRobot.LogoFacingDirection.UP;
|
||||||
|
RevHubOrientationOnRobot.UsbFacingDirection usbDirection = RevHubOrientationOnRobot.UsbFacingDirection.FORWARD;
|
||||||
|
|
||||||
|
RevHubOrientationOnRobot orientationOnRobot = new RevHubOrientationOnRobot(logoDirection, usbDirection);
|
||||||
|
|
||||||
|
// Now initialize the IMU with this mounting orientation
|
||||||
|
// Note: if you choose two conflicting directions, this initialization will cause a code exception.
|
||||||
|
imu.initialize(new IMU.Parameters(orientationOnRobot));
|
||||||
|
|
||||||
|
// Loop and update the dashboard
|
||||||
|
while (!isStopRequested()) {
|
||||||
|
|
||||||
|
telemetry.addData("Hub orientation", "Logo=%s USB=%s\n ", logoDirection, usbDirection);
|
||||||
|
|
||||||
|
// Check to see if heading reset is requested
|
||||||
|
if (gamepad1.y) {
|
||||||
|
telemetry.addData("Yaw", "Resetting\n");
|
||||||
|
imu.resetYaw();
|
||||||
|
} else {
|
||||||
|
telemetry.addData("Yaw", "Press Y (triangle) on Gamepad to reset\n");
|
||||||
|
}
|
||||||
|
|
||||||
|
// Retrieve Rotational Angles and Velocities
|
||||||
|
YawPitchRollAngles orientation = imu.getRobotYawPitchRollAngles();
|
||||||
|
AngularVelocity angularVelocity = imu.getRobotAngularVelocity(AngleUnit.DEGREES);
|
||||||
|
|
||||||
|
telemetry.addData("Yaw (Z)", "%.2f Deg. (Heading)", orientation.getYaw(AngleUnit.DEGREES));
|
||||||
|
telemetry.addData("Pitch (X)", "%.2f Deg.", orientation.getPitch(AngleUnit.DEGREES));
|
||||||
|
telemetry.addData("Roll (Y)", "%.2f Deg.\n", orientation.getRoll(AngleUnit.DEGREES));
|
||||||
|
telemetry.addData("Yaw (Z) velocity", "%.2f Deg/Sec", angularVelocity.zRotationRate);
|
||||||
|
telemetry.addData("Pitch (X) velocity", "%.2f Deg/Sec", angularVelocity.xRotationRate);
|
||||||
|
telemetry.addData("Roll (Y) velocity", "%.2f Deg/Sec", angularVelocity.yRotationRate);
|
||||||
|
telemetry.update();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
@ -0,0 +1,189 @@
|
|||||||
|
/* Copyright (c) 2021 FIRST. All rights reserved.
|
||||||
|
*
|
||||||
|
* Redistribution and use in source and binary forms, with or without modification,
|
||||||
|
* are permitted (subject to the limitations in the disclaimer below) provided that
|
||||||
|
* the following conditions are met:
|
||||||
|
*
|
||||||
|
* Redistributions of source code must retain the above copyright notice, this list
|
||||||
|
* of conditions and the following disclaimer.
|
||||||
|
*
|
||||||
|
* Redistributions in binary form must reproduce the above copyright notice, this
|
||||||
|
* list of conditions and the following disclaimer in the documentation and/or
|
||||||
|
* other materials provided with the distribution.
|
||||||
|
*
|
||||||
|
* Neither the name of FIRST nor the names of its contributors may be used to endorse or
|
||||||
|
* promote products derived from this software without specific prior written permission.
|
||||||
|
*
|
||||||
|
* NO EXPRESS OR IMPLIED LICENSES TO ANY PARTY'S PATENT RIGHTS ARE GRANTED BY THIS
|
||||||
|
* LICENSE. THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
|
||||||
|
* "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO,
|
||||||
|
* THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
|
||||||
|
* ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE
|
||||||
|
* FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
|
||||||
|
* DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
|
||||||
|
* SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER
|
||||||
|
* CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY,
|
||||||
|
* OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
|
||||||
|
* OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
||||||
|
*/
|
||||||
|
|
||||||
|
package org.firstinspires.ftc.teamcode;
|
||||||
|
|
||||||
|
import static org.firstinspires.ftc.teamcode.cometbots.Constants.*;
|
||||||
|
|
||||||
|
import com.qualcomm.robotcore.eventloop.opmode.Disabled;
|
||||||
|
import com.qualcomm.robotcore.eventloop.opmode.LinearOpMode;
|
||||||
|
import com.qualcomm.robotcore.eventloop.opmode.TeleOp;
|
||||||
|
import com.qualcomm.robotcore.hardware.DcMotor;
|
||||||
|
import com.qualcomm.robotcore.hardware.DcMotorSimple;
|
||||||
|
import com.qualcomm.robotcore.hardware.Gamepad;
|
||||||
|
import com.qualcomm.robotcore.util.ElapsedTime;
|
||||||
|
|
||||||
|
import org.firstinspires.ftc.teamcode.cometbots.MovementLibrary;
|
||||||
|
|
||||||
|
/*
|
||||||
|
* This file contains an example of a Linear "OpMode".
|
||||||
|
* An OpMode is a 'program' that runs in either the autonomous or the teleop period of an FTC match.
|
||||||
|
* The names of OpModes appear on the menu of the FTC Driver Station.
|
||||||
|
* When a selection is made from the menu, the corresponding OpMode is executed.
|
||||||
|
*
|
||||||
|
* This particular OpMode illustrates driving a 4-motor Omni-Directional (or Holonomic) robot.
|
||||||
|
* This code will work with either a Mecanum-Drive or an X-Drive train.
|
||||||
|
* Both of these drives are illustrated at https://gm0.org/en/latest/docs/robot-design/drivetrains/holonomic.html
|
||||||
|
* Note that a Mecanum drive must display an X roller-pattern when viewed from above.
|
||||||
|
*
|
||||||
|
* Also note that it is critical to set the correct rotation direction for each motor. See details below.
|
||||||
|
*
|
||||||
|
* Holonomic drives provide the ability for the robot to move in three axes (directions) simultaneously.
|
||||||
|
* Each motion axis is controlled by one Joystick axis.
|
||||||
|
*
|
||||||
|
* 1) Axial: Driving forward and backward Left-joystick Forward/Backward
|
||||||
|
* 2) Lateral: Strafing right and left Left-joystick Right and Left
|
||||||
|
* 3) Yaw: Rotating Clockwise and counter clockwise Right-joystick Right and Left
|
||||||
|
*
|
||||||
|
* This code is written assuming that the right-side motors need to be reversed for the robot to drive forward.
|
||||||
|
* When you first test your robot, if it moves backward when you push the left stick forward, then you must flip
|
||||||
|
* the direction of all 4 motors (see code below).
|
||||||
|
*
|
||||||
|
* Use Android Studio to Copy this Class, and Paste it into your team's code folder with a new name.
|
||||||
|
* Remove or comment out the @Disabled line to add this OpMode to the Driver Station OpMode list
|
||||||
|
*/
|
||||||
|
|
||||||
|
@TeleOp(name="CometBots Linear OpMode Demo", group = "[14493] CometBots")
|
||||||
|
public class CometBotsLinearOpMode extends LinearOpMode {
|
||||||
|
|
||||||
|
// Declare OpMode members for each of the 4 motors.
|
||||||
|
private ElapsedTime runtime = new ElapsedTime();
|
||||||
|
private DcMotor rhw = null;
|
||||||
|
private DcMotor rlw = null;
|
||||||
|
private DcMotor lhw = null;
|
||||||
|
private DcMotor llw = null;
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void runOpMode() {
|
||||||
|
|
||||||
|
// Button A counter to cycle speeds
|
||||||
|
int cntBtnA = 1;
|
||||||
|
|
||||||
|
// Initialize the hardware variables. Note that the strings used here must correspond
|
||||||
|
// to the names assigned during the robot configuration step on the DS or RC devices.
|
||||||
|
rhw = hardwareMap.get(DcMotor.class, RT_FRONT);
|
||||||
|
rlw = hardwareMap.get(DcMotor.class, RT_BACK);
|
||||||
|
lhw = hardwareMap.get(DcMotor.class, LT_FRONT);
|
||||||
|
llw = hardwareMap.get(DcMotor.class, LT_BACK);
|
||||||
|
|
||||||
|
/*
|
||||||
|
Put all motors in an array for easier handling
|
||||||
|
|
||||||
|
NOTE: Placement of motor in array is important especially when dealing with functions
|
||||||
|
that set direction
|
||||||
|
*/
|
||||||
|
DcMotor[] motors = {rhw, rlw, lhw, llw};
|
||||||
|
|
||||||
|
/*
|
||||||
|
Initialize all wheels forward using motors a
|
||||||
|
*/
|
||||||
|
MovementLibrary.setDirectionForward(motors);
|
||||||
|
|
||||||
|
telemetry.addData("Status", "Initialized");
|
||||||
|
telemetry.update();
|
||||||
|
|
||||||
|
waitForStart();
|
||||||
|
runtime.reset();
|
||||||
|
|
||||||
|
// run until the end of the match (driver presses STOP)
|
||||||
|
while (opModeIsActive()) {
|
||||||
|
double max;
|
||||||
|
|
||||||
|
// POV Mode uses left joystick to go forward & strafe, and right joystick to rotate.
|
||||||
|
double axial = -gamepad1.left_stick_y; // Note: pushing stick forward gives negative value
|
||||||
|
double lateral = gamepad1.left_stick_x;
|
||||||
|
double yaw = gamepad1.right_stick_x;
|
||||||
|
|
||||||
|
// Combine the joystick requests for each axis-motion to determine each wheel's power.
|
||||||
|
// Set up a variable for each drive wheel to save the power level for telemetry.
|
||||||
|
double lftFrntPwr = axial + lateral + yaw;
|
||||||
|
double rtFrntPwr = axial - lateral - yaw;
|
||||||
|
double ltBckPwr = axial - lateral + yaw;
|
||||||
|
double rtBckPwr = axial + lateral - yaw;
|
||||||
|
|
||||||
|
// Normalize the values so no wheel power exceeds 100%
|
||||||
|
// This ensures that the robot maintains the desired motion.
|
||||||
|
max = Math.max(Math.abs(lftFrntPwr), Math.abs(rtFrntPwr));
|
||||||
|
max = Math.max(max, Math.abs(ltBckPwr));
|
||||||
|
max = Math.max(max, Math.abs(rtBckPwr));
|
||||||
|
|
||||||
|
if (max > 1.0) {
|
||||||
|
lftFrntPwr /= max;
|
||||||
|
rtFrntPwr /= max;
|
||||||
|
ltBckPwr /= max;
|
||||||
|
rtBckPwr /= max;
|
||||||
|
}
|
||||||
|
|
||||||
|
// This is test code:
|
||||||
|
//
|
||||||
|
// Uncomment the following code to test your motor directions.
|
||||||
|
// Each button should make the corresponding motor run FORWARD.
|
||||||
|
// 1) First get all the motors to take to correct positions on the robot
|
||||||
|
// by adjusting your Robot Configuration if necessary.
|
||||||
|
// 2) Then make sure they run in the correct direction by modifying the
|
||||||
|
// the setDirection() calls above.
|
||||||
|
// Once the correct motors move in the correct direction re-comment this code.
|
||||||
|
|
||||||
|
/*
|
||||||
|
lftFrntPwr = gamepad1.x ? 1.0 : 0.0; // X gamepad
|
||||||
|
ltBckPwr = gamepad1.a ? 1.0 : 0.0; // A gamepad
|
||||||
|
rtFrntPwr = gamepad1.y ? 1.0 : 0.0; // Y gamepad
|
||||||
|
rtBckPwr = gamepad1.b ? 1.0 : 0.0; // B gamepad
|
||||||
|
*/
|
||||||
|
|
||||||
|
lhw.setPower(lftFrntPwr/cntBtnA);
|
||||||
|
rhw.setPower(rtFrntPwr/cntBtnA);
|
||||||
|
llw.setPower(ltBckPwr/cntBtnA);
|
||||||
|
rlw.setPower(rtBckPwr/cntBtnA);
|
||||||
|
|
||||||
|
// Send calculated power to wheels
|
||||||
|
if(gamepad1.left_bumper) {
|
||||||
|
sleep(175);
|
||||||
|
cntBtnA--;
|
||||||
|
if(cntBtnA < 0) {
|
||||||
|
cntBtnA = 0;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if(gamepad1.right_bumper) {
|
||||||
|
sleep(175);
|
||||||
|
cntBtnA++;
|
||||||
|
if(cntBtnA > 4) {
|
||||||
|
cntBtnA = 4;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Show the elapsed game time and wheel power.
|
||||||
|
telemetry.addData("Status", "Run Time: " + runtime.toString());
|
||||||
|
telemetry.addData("Front left/Right", "%4.2f, %4.2f", lftFrntPwr, rtFrntPwr);
|
||||||
|
telemetry.addData("Back left/Right", "%4.2f, %4.2f", ltBckPwr, rtBckPwr);
|
||||||
|
telemetry.addData("Power Mode", cntBtnA);
|
||||||
|
telemetry.update();
|
||||||
|
}
|
||||||
|
}}
|
@ -0,0 +1,132 @@
|
|||||||
|
/* Copyright (c) 2021 FIRST. All rights reserved.
|
||||||
|
*
|
||||||
|
* Redistribution and use in source and binary forms, with or without modification,
|
||||||
|
* are permitted (subject to the limitations in the disclaimer below) provided that
|
||||||
|
* the following conditions are met:
|
||||||
|
*
|
||||||
|
* Redistributions of source code must retain the above copyright notice, this list
|
||||||
|
* of conditions and the following disclaimer.
|
||||||
|
*
|
||||||
|
* Redistributions in binary form must reproduce the above copyright notice, this
|
||||||
|
* list of conditions and the following disclaimer in the documentation and/or
|
||||||
|
* other materials provided with the distribution.
|
||||||
|
*
|
||||||
|
* Neither the name of FIRST nor the names of its contributors may be used to endorse or
|
||||||
|
* promote products derived from this software without specific prior written permission.
|
||||||
|
*
|
||||||
|
* NO EXPRESS OR IMPLIED LICENSES TO ANY PARTY'S PATENT RIGHTS ARE GRANTED BY THIS
|
||||||
|
* LICENSE. THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
|
||||||
|
* "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO,
|
||||||
|
* THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
|
||||||
|
* ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE
|
||||||
|
* FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
|
||||||
|
* DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
|
||||||
|
* SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER
|
||||||
|
* CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY,
|
||||||
|
* OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
|
||||||
|
* OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
||||||
|
*/
|
||||||
|
|
||||||
|
package org.firstinspires.ftc.teamcode;
|
||||||
|
|
||||||
|
import static org.firstinspires.ftc.teamcode.cometbots.Constants.LT_BACK;
|
||||||
|
import static org.firstinspires.ftc.teamcode.cometbots.Constants.LT_FRONT;
|
||||||
|
import static org.firstinspires.ftc.teamcode.cometbots.Constants.RT_BACK;
|
||||||
|
import static org.firstinspires.ftc.teamcode.cometbots.Constants.RT_FRONT;
|
||||||
|
|
||||||
|
import com.acmerobotics.roadrunner.Pose2d;
|
||||||
|
import com.acmerobotics.roadrunner.PoseVelocity2d;
|
||||||
|
import com.acmerobotics.roadrunner.Vector2d;
|
||||||
|
import com.qualcomm.robotcore.eventloop.opmode.LinearOpMode;
|
||||||
|
import com.qualcomm.robotcore.eventloop.opmode.TeleOp;
|
||||||
|
import com.qualcomm.robotcore.hardware.DcMotor;
|
||||||
|
import com.qualcomm.robotcore.util.ElapsedTime;
|
||||||
|
|
||||||
|
import org.firstinspires.ftc.teamcode.cometbots.MovementLibrary;
|
||||||
|
|
||||||
|
/*
|
||||||
|
* This file contains an example of a Linear "OpMode".
|
||||||
|
* An OpMode is a 'program' that runs in either the autonomous or the teleop period of an FTC match.
|
||||||
|
* The names of OpModes appear on the menu of the FTC Driver Station.
|
||||||
|
* When a selection is made from the menu, the corresponding OpMode is executed.
|
||||||
|
*
|
||||||
|
* This particular OpMode illustrates driving a 4-motor Omni-Directional (or Holonomic) robot.
|
||||||
|
* This code will work with either a Mecanum-Drive or an X-Drive train.
|
||||||
|
* Both of these drives are illustrated at https://gm0.org/en/latest/docs/robot-design/drivetrains/holonomic.html
|
||||||
|
* Note that a Mecanum drive must display an X roller-pattern when viewed from above.
|
||||||
|
*
|
||||||
|
* Also note that it is critical to set the correct rotation direction for each motor. See details below.
|
||||||
|
*
|
||||||
|
* Holonomic drives provide the ability for the robot to move in three axes (directions) simultaneously.
|
||||||
|
* Each motion axis is controlled by one Joystick axis.
|
||||||
|
*
|
||||||
|
* 1) Axial: Driving forward and backward Left-joystick Forward/Backward
|
||||||
|
* 2) Lateral: Strafing right and left Left-joystick Right and Left
|
||||||
|
* 3) Yaw: Rotating Clockwise and counter clockwise Right-joystick Right and Left
|
||||||
|
*
|
||||||
|
* This code is written assuming that the right-side motors need to be reversed for the robot to drive forward.
|
||||||
|
* When you first test your robot, if it moves backward when you push the left stick forward, then you must flip
|
||||||
|
* the direction of all 4 motors (see code below).
|
||||||
|
*
|
||||||
|
* Use Android Studio to Copy this Class, and Paste it into your team's code folder with a new name.
|
||||||
|
* Remove or comment out the @Disabled line to add this OpMode to the Driver Station OpMode list
|
||||||
|
*/
|
||||||
|
|
||||||
|
@TeleOp(name="CometBots Mecanum Drive Demo", group = "[14493] CometBots")
|
||||||
|
public class CometBotsMecanumDriveMode extends LinearOpMode {
|
||||||
|
|
||||||
|
// Declare OpMode members for each of the 4 motors.
|
||||||
|
private ElapsedTime runtime = new ElapsedTime();
|
||||||
|
private DcMotor rhw = null;
|
||||||
|
private DcMotor rlw = null;
|
||||||
|
private DcMotor lhw = null;
|
||||||
|
private DcMotor llw = null;
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void runOpMode() {
|
||||||
|
|
||||||
|
MecanumDrive drive = new MecanumDrive(hardwareMap, new Pose2d(0,0,Math.toRadians(90)));
|
||||||
|
|
||||||
|
// Button A counter to cycle speeds
|
||||||
|
int cntBtnA = 1;
|
||||||
|
|
||||||
|
// Initialize the hardware variables. Note that the strings used here must correspond
|
||||||
|
// to the names assigned during the robot configuration step on the DS or RC devices.
|
||||||
|
rhw = hardwareMap.get(DcMotor.class, RT_FRONT);
|
||||||
|
rlw = hardwareMap.get(DcMotor.class, RT_BACK);
|
||||||
|
lhw = hardwareMap.get(DcMotor.class, LT_FRONT);
|
||||||
|
llw = hardwareMap.get(DcMotor.class, LT_BACK);
|
||||||
|
|
||||||
|
/*
|
||||||
|
Put all motors in an array for easier handling
|
||||||
|
|
||||||
|
NOTE: Placement of motor in array is important especially when dealing with functions
|
||||||
|
that set direction
|
||||||
|
*/
|
||||||
|
DcMotor[] motors = {rhw, rlw, lhw, llw};
|
||||||
|
|
||||||
|
/*
|
||||||
|
Initialize all wheels forward using motors a
|
||||||
|
*/
|
||||||
|
MovementLibrary.setDirectionForward(motors);
|
||||||
|
|
||||||
|
telemetry.addData("Status", "Initialized");
|
||||||
|
telemetry.update();
|
||||||
|
|
||||||
|
waitForStart();
|
||||||
|
runtime.reset();
|
||||||
|
|
||||||
|
// run until the end of the match (driver presses STOP)
|
||||||
|
while (opModeIsActive()) {
|
||||||
|
drive.setDrivePowers(
|
||||||
|
new PoseVelocity2d(
|
||||||
|
new Vector2d(
|
||||||
|
gamepad1.right_stick_x,
|
||||||
|
gamepad1.right_stick_y
|
||||||
|
),
|
||||||
|
-gamepad1.right_stick_x
|
||||||
|
)
|
||||||
|
);
|
||||||
|
drive.updatePoseEstimate();
|
||||||
|
}
|
||||||
|
}}
|
@ -0,0 +1,46 @@
|
|||||||
|
package org.firstinspires.ftc.teamcode;
|
||||||
|
|
||||||
|
import com.qualcomm.robotcore.eventloop.opmode.Autonomous;
|
||||||
|
import com.qualcomm.robotcore.eventloop.opmode.Disabled;
|
||||||
|
import com.qualcomm.robotcore.eventloop.opmode.LinearOpMode;
|
||||||
|
|
||||||
|
import org.firstinspires.ftc.robotcore.external.hardware.camera.WebcamName;
|
||||||
|
import org.openftc.easyopencv.OpenCvCamera;
|
||||||
|
import org.openftc.easyopencv.OpenCvCameraFactory;
|
||||||
|
import org.openftc.easyopencv.OpenCvCameraRotation;
|
||||||
|
|
||||||
|
@Disabled
|
||||||
|
@Autonomous(name="DevApp - OpenCV Demo", group="[14493] CometBots")
|
||||||
|
public class CometBotsOpenCVDemo extends LinearOpMode {
|
||||||
|
|
||||||
|
private OpenCvCamera controlHubCam; // Use OpenCvCamera class from FTC SDK
|
||||||
|
private static final int CAMERA_WIDTH = 1280; // width of wanted camera resolution
|
||||||
|
private static final int CAMERA_HEIGHT = 720; // height of wanted camera resolution
|
||||||
|
|
||||||
|
// Calculate the distance using the formula
|
||||||
|
public static final double objectWidthInRealWorldUnits = 3.75; // Replace with the actual width of the object in real-world units
|
||||||
|
public static final double focalLength = 728; // Replace with the focal length of the camera in pixels
|
||||||
|
|
||||||
|
private void initOpenCV() {
|
||||||
|
|
||||||
|
// Create an instance of the camera
|
||||||
|
int cameraMonitorViewId = hardwareMap.appContext.getResources().getIdentifier(
|
||||||
|
"cameraMonitorViewId", "id", hardwareMap.appContext.getPackageName());
|
||||||
|
|
||||||
|
// Use OpenCvCameraFactory class from FTC SDK to create camera instance
|
||||||
|
controlHubCam = OpenCvCameraFactory.getInstance().createWebcam(
|
||||||
|
hardwareMap.get(WebcamName.class, "watch out"), cameraMonitorViewId);
|
||||||
|
|
||||||
|
// controlHubCam.setPipeline(new YellowBlobDetectionPipeline());
|
||||||
|
|
||||||
|
controlHubCam.openCameraDevice();
|
||||||
|
controlHubCam.startStreaming(CAMERA_WIDTH, CAMERA_HEIGHT, OpenCvCameraRotation.UPRIGHT);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void runOpMode() throws InterruptedException {
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
}
|
||||||
|
}
|
@ -0,0 +1,100 @@
|
|||||||
|
package org.firstinspires.ftc.teamcode;
|
||||||
|
|
||||||
|
import static org.firstinspires.ftc.teamcode.cometbots.Constants.*;
|
||||||
|
|
||||||
|
import com.qualcomm.robotcore.eventloop.opmode.Autonomous;
|
||||||
|
import com.qualcomm.robotcore.eventloop.opmode.Disabled;
|
||||||
|
import com.qualcomm.robotcore.eventloop.opmode.LinearOpMode;
|
||||||
|
import com.qualcomm.robotcore.hardware.ColorSensor;
|
||||||
|
import com.qualcomm.robotcore.hardware.DcMotor;
|
||||||
|
import com.qualcomm.robotcore.hardware.DistanceSensor;
|
||||||
|
import com.qualcomm.robotcore.hardware.TouchSensor;
|
||||||
|
|
||||||
|
import org.firstinspires.ftc.robotcore.external.navigation.DistanceUnit;
|
||||||
|
import org.firstinspires.ftc.teamcode.cometbots.MovementLibrary;
|
||||||
|
|
||||||
|
@Disabled
|
||||||
|
@Autonomous(name = "Development Class", group = "[14493] CometBots")
|
||||||
|
public class CometBotsProject extends LinearOpMode {
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void runOpMode() throws InterruptedException {
|
||||||
|
|
||||||
|
DcMotor rhw = hardwareMap.get(DcMotor.class, "right hand wheel");
|
||||||
|
DcMotor rlw = hardwareMap.get(DcMotor.class, "right leg wheel");
|
||||||
|
DcMotor lhw = hardwareMap.get(DcMotor.class, "left hand wheel");
|
||||||
|
DcMotor llw = hardwareMap.get(DcMotor.class, "left leg wheel");
|
||||||
|
TouchSensor touchSensor = hardwareMap.get(TouchSensor.class, "touchsensor");
|
||||||
|
DistanceSensor distanceSensor = hardwareMap.get(DistanceSensor.class, "distancesensor");
|
||||||
|
ColorSensor colorSensor = hardwareMap.get(ColorSensor.class, "colorsensor");
|
||||||
|
|
||||||
|
/*
|
||||||
|
Put all motors in an array for easier handling
|
||||||
|
|
||||||
|
NOTE: Placement of motor in array is important
|
||||||
|
especially when dealing with functions that
|
||||||
|
set direction
|
||||||
|
*/
|
||||||
|
DcMotor[] motors = {rhw, rlw, lhw, llw};
|
||||||
|
|
||||||
|
/*
|
||||||
|
Initialize all wheels forward using motors array
|
||||||
|
*/
|
||||||
|
// MovementLibrary.setDirectionForward(motors);
|
||||||
|
MovementLibrary.setLeftSideBackward(motors);
|
||||||
|
MovementLibrary.setRightSideForward(motors);
|
||||||
|
|
||||||
|
/*
|
||||||
|
Capture color readout from color sensor
|
||||||
|
*/
|
||||||
|
telemetry.addData("Color Sensor [Red]", colorSensor.red());
|
||||||
|
telemetry.addData("Color Sensor [Green]", colorSensor.green());
|
||||||
|
telemetry.addData("Color Sensor [Blue]", colorSensor.blue());
|
||||||
|
telemetry.addData("Color Sensor [Alpha]", colorSensor.alpha());
|
||||||
|
|
||||||
|
/*
|
||||||
|
Capture distance readout from distance sensor
|
||||||
|
*/
|
||||||
|
telemetry.addData("Distance Sensor [inches]", distanceSensor.getDistance(DistanceUnit.INCH));
|
||||||
|
|
||||||
|
/*
|
||||||
|
Capture state when touch sensor is pressed
|
||||||
|
*/
|
||||||
|
if (touchSensor.isPressed()) {
|
||||||
|
telemetry.speak("Touch sensor is pressed");
|
||||||
|
telemetry.addData("Touch sensor pressed?", touchSensor.isPressed());
|
||||||
|
}
|
||||||
|
|
||||||
|
telemetry.update();
|
||||||
|
|
||||||
|
/*
|
||||||
|
Wait for the driver to hit START in the driver hub
|
||||||
|
*/
|
||||||
|
waitForStart();
|
||||||
|
|
||||||
|
while (opModeIsActive()) {
|
||||||
|
|
||||||
|
/*
|
||||||
|
Initialize wheel direction with base speed
|
||||||
|
*/
|
||||||
|
MovementLibrary.setSpeed(motors, CB_BASE_SPEED);
|
||||||
|
|
||||||
|
|
||||||
|
/*
|
||||||
|
Distance traveled in inches
|
||||||
|
*/
|
||||||
|
double distanceInInches = distanceSensor.getDistance(DistanceUnit.INCH);
|
||||||
|
telemetry.addData("Distance traveled (inches)", "%.2f", distanceInInches);
|
||||||
|
|
||||||
|
/*
|
||||||
|
Speed up when going over an object that is blue-ish / green
|
||||||
|
*/
|
||||||
|
if (colorSensor.blue() > 800 && colorSensor.green() > 1100) {
|
||||||
|
telemetry.speak("Green speed boost");
|
||||||
|
MovementLibrary.setSpeed(motors, CB_BASE_SPEED * 2.0);
|
||||||
|
} else {
|
||||||
|
MovementLibrary.setSpeed(motors, CB_BASE_SPEED);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
@ -0,0 +1,111 @@
|
|||||||
|
package org.firstinspires.ftc.teamcode;
|
||||||
|
|
||||||
|
import com.acmerobotics.dashboard.FtcDashboard;
|
||||||
|
import com.qualcomm.robotcore.eventloop.opmode.Autonomous;
|
||||||
|
import com.qualcomm.robotcore.eventloop.opmode.LinearOpMode;
|
||||||
|
import com.qualcomm.robotcore.hardware.ColorSensor;
|
||||||
|
import com.qualcomm.robotcore.hardware.DcMotor;
|
||||||
|
import com.qualcomm.robotcore.hardware.DistanceSensor;
|
||||||
|
import com.qualcomm.robotcore.hardware.TouchSensor;
|
||||||
|
|
||||||
|
import org.firstinspires.ftc.robotcore.external.hardware.camera.WebcamName;
|
||||||
|
import org.firstinspires.ftc.robotcore.external.navigation.DistanceUnit;
|
||||||
|
import org.firstinspires.ftc.teamcode.cometbots.MovementLibrary;
|
||||||
|
import org.openftc.easyopencv.OpenCvCamera;
|
||||||
|
import org.openftc.easyopencv.OpenCvCameraFactory;
|
||||||
|
import org.openftc.easyopencv.OpenCvCameraRotation;
|
||||||
|
|
||||||
|
@Autonomous(name = "CometBot Sensor Demo", group = "[14493] CometBots")
|
||||||
|
public class CometBotsSensorDemo extends LinearOpMode {
|
||||||
|
|
||||||
|
private OpenCvCamera controlHubCam; // Use OpenCvCamera class from FTC SDK
|
||||||
|
private static final int CAMERA_WIDTH = 1280; // width of wanted camera resolution
|
||||||
|
private static final int CAMERA_HEIGHT = 720; // height of wanted camera resolution
|
||||||
|
|
||||||
|
// Calculate the distance using the formula
|
||||||
|
public static final double objectWidthInRealWorldUnits = 3.75; // Replace with the actual width of the object in real-world units
|
||||||
|
public static final double focalLength = 728; // Replace with the focal length of the camera in pixels
|
||||||
|
|
||||||
|
private DcMotor rhw = null;
|
||||||
|
private DcMotor rlw = null;
|
||||||
|
private DcMotor lhw = null;
|
||||||
|
private DcMotor llw = null;
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void runOpMode() throws InterruptedException {
|
||||||
|
|
||||||
|
// Initialize the hardware variables. Note that the strings used here must correspond
|
||||||
|
// to the names assigned during the robot configuration step on the DS or RC devices.
|
||||||
|
rhw = hardwareMap.get(DcMotor.class, "right hand wheel");
|
||||||
|
rlw = hardwareMap.get(DcMotor.class, "right leg wheel");
|
||||||
|
lhw = hardwareMap.get(DcMotor.class, "left hand wheel");
|
||||||
|
llw = hardwareMap.get(DcMotor.class, "left leg wheel");
|
||||||
|
|
||||||
|
/*
|
||||||
|
Put all motors in an array for easier handling
|
||||||
|
|
||||||
|
NOTE: Placement of motor in array is important especially when dealing with functions
|
||||||
|
that set direction
|
||||||
|
*/
|
||||||
|
DcMotor[] motors = {rhw, rlw, lhw, llw};
|
||||||
|
|
||||||
|
// Create an instance of the camera
|
||||||
|
int cameraMonitorViewId = hardwareMap.appContext.getResources().getIdentifier(
|
||||||
|
"cameraMonitorViewId", "id", hardwareMap.appContext.getPackageName());
|
||||||
|
|
||||||
|
// Use OpenCvCameraFactory class from FTC SDK to create camera instance
|
||||||
|
controlHubCam = OpenCvCameraFactory.getInstance().createWebcam(
|
||||||
|
hardwareMap.get(WebcamName.class, "watch out"), cameraMonitorViewId);
|
||||||
|
|
||||||
|
TouchSensor touchSensor = hardwareMap.get(TouchSensor.class, "touchsensor");
|
||||||
|
DistanceSensor distanceSensor = hardwareMap.get(DistanceSensor.class, "distancesensor");
|
||||||
|
ColorSensor colorSensor = hardwareMap.get(ColorSensor.class, "colorsensor");
|
||||||
|
|
||||||
|
controlHubCam.openCameraDevice();
|
||||||
|
controlHubCam.startStreaming(CAMERA_WIDTH, CAMERA_HEIGHT, OpenCvCameraRotation.UPRIGHT);
|
||||||
|
|
||||||
|
FtcDashboard.getInstance().startCameraStream(controlHubCam, 0);
|
||||||
|
|
||||||
|
/*
|
||||||
|
Wait for the driver to hit START in the driver hub
|
||||||
|
*/
|
||||||
|
waitForStart();
|
||||||
|
|
||||||
|
while (opModeIsActive()) {
|
||||||
|
|
||||||
|
/*
|
||||||
|
Capture color readout from color sensor
|
||||||
|
*/
|
||||||
|
telemetry.addData("Color Sensor [Red]", colorSensor.red());
|
||||||
|
telemetry.addData("Color Sensor [Green]", colorSensor.green());
|
||||||
|
telemetry.addData("Color Sensor [Blue]", colorSensor.blue());
|
||||||
|
telemetry.addData("Color Sensor [Alpha]", colorSensor.alpha());
|
||||||
|
|
||||||
|
/*
|
||||||
|
Capture distance readout from distance sensor
|
||||||
|
*/
|
||||||
|
telemetry.addData("Distance Sensor [inches]", distanceSensor.getDistance(DistanceUnit.INCH));
|
||||||
|
|
||||||
|
/*
|
||||||
|
Capture state when touch sensor is pressed
|
||||||
|
*/
|
||||||
|
telemetry.addData("Touch sensor pressed?", touchSensor.isPressed());
|
||||||
|
if (touchSensor.isPressed()) {
|
||||||
|
telemetry.speak("Touch sensor is pressed");
|
||||||
|
}
|
||||||
|
|
||||||
|
if (distanceSensor.getDistance(DistanceUnit.INCH) < 3) {
|
||||||
|
telemetry.speak("Warning, unknown object near robot");
|
||||||
|
MovementLibrary.setDirectionRight(motors);
|
||||||
|
MovementLibrary.setSpeed(motors,0.25);
|
||||||
|
sleep(1500);
|
||||||
|
MovementLibrary.setSpeed(motors,0);
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
Update stats continually
|
||||||
|
*/
|
||||||
|
telemetry.update();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
@ -0,0 +1,141 @@
|
|||||||
|
/* Copyright (c) 2019 FIRST. All rights reserved.
|
||||||
|
*
|
||||||
|
* Redistribution and use in source and binary forms, with or without modification,
|
||||||
|
* are permitted (subject to the limitations in the disclaimer below) provided that
|
||||||
|
* the following conditions are met:
|
||||||
|
*
|
||||||
|
* Redistributions of source code must retain the above copyright notice, this list
|
||||||
|
* of conditions and the following disclaimer.
|
||||||
|
*
|
||||||
|
* Redistributions in binary form must reproduce the above copyright notice, this
|
||||||
|
* list of conditions and the following disclaimer in the documentation and/or
|
||||||
|
* other materials provided with the distribution.
|
||||||
|
*
|
||||||
|
* Neither the name of FIRST nor the names of its contributors may be used to endorse or
|
||||||
|
* promote products derived from this software without specific prior written permission.
|
||||||
|
*
|
||||||
|
* NO EXPRESS OR IMPLIED LICENSES TO ANY PARTY'S PATENT RIGHTS ARE GRANTED BY THIS
|
||||||
|
* LICENSE. THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
|
||||||
|
* "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO,
|
||||||
|
* THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
|
||||||
|
* ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE
|
||||||
|
* FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
|
||||||
|
* DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
|
||||||
|
* SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER
|
||||||
|
* CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY,
|
||||||
|
* OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
|
||||||
|
* OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
|
||||||
|
*/
|
||||||
|
|
||||||
|
package org.firstinspires.ftc.teamcode;
|
||||||
|
|
||||||
|
import com.qualcomm.robotcore.eventloop.opmode.LinearOpMode;
|
||||||
|
import com.qualcomm.robotcore.eventloop.opmode.TeleOp;
|
||||||
|
|
||||||
|
import org.firstinspires.ftc.robotcore.external.hardware.camera.BuiltinCameraDirection;
|
||||||
|
import org.firstinspires.ftc.robotcore.external.hardware.camera.WebcamName;
|
||||||
|
import org.firstinspires.ftc.robotcore.external.tfod.Recognition;
|
||||||
|
import org.firstinspires.ftc.vision.VisionPortal;
|
||||||
|
import org.firstinspires.ftc.vision.tfod.TfodProcessor;
|
||||||
|
|
||||||
|
import java.util.List;
|
||||||
|
|
||||||
|
/*
|
||||||
|
* This OpMode illustrates the basics of TensorFlow Object Detection, using
|
||||||
|
* the easiest way.
|
||||||
|
*
|
||||||
|
* Use Android Studio to Copy this Class, and Paste it into your team's code folder with a new name.
|
||||||
|
* Remove or comment out the @Disabled line to add this OpMode to the Driver Station OpMode list.
|
||||||
|
*/
|
||||||
|
@TeleOp(name = "CometBots TensorFlow Demo (HDMI)", group = "[14493] CometBots")
|
||||||
|
public class CometBotsTensorFlowDemo extends LinearOpMode {
|
||||||
|
|
||||||
|
private static final boolean USE_WEBCAM = true; // true for webcam, false for phone camera
|
||||||
|
|
||||||
|
/**
|
||||||
|
* The variable to store our instance of the TensorFlow Object Detection processor.
|
||||||
|
*/
|
||||||
|
private TfodProcessor tfod;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* The variable to store our instance of the vision portal.
|
||||||
|
*/
|
||||||
|
private VisionPortal visionPortal;
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void runOpMode() {
|
||||||
|
|
||||||
|
initTfod();
|
||||||
|
|
||||||
|
// Wait for the DS start button to be touched.
|
||||||
|
telemetry.addData("DS preview on/off", "3 dots, Camera Stream");
|
||||||
|
telemetry.addData(">", "Touch Play to start OpMode");
|
||||||
|
telemetry.update();
|
||||||
|
waitForStart();
|
||||||
|
|
||||||
|
if (opModeIsActive()) {
|
||||||
|
while (opModeIsActive()) {
|
||||||
|
|
||||||
|
telemetryTfod();
|
||||||
|
|
||||||
|
// Push telemetry to the Driver Station.
|
||||||
|
telemetry.update();
|
||||||
|
|
||||||
|
// Save CPU resources; can resume streaming when needed.
|
||||||
|
if (gamepad1.dpad_down) {
|
||||||
|
visionPortal.stopStreaming();
|
||||||
|
} else if (gamepad1.dpad_up) {
|
||||||
|
visionPortal.resumeStreaming();
|
||||||
|
}
|
||||||
|
|
||||||
|
// Share the CPU.
|
||||||
|
sleep(20);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Save more CPU resources when camera is no longer needed.
|
||||||
|
visionPortal.close();
|
||||||
|
|
||||||
|
} // end runOpMode()
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Initialize the TensorFlow Object Detection processor.
|
||||||
|
*/
|
||||||
|
private void initTfod() {
|
||||||
|
|
||||||
|
// Create the TensorFlow processor the easy way.
|
||||||
|
tfod = TfodProcessor.easyCreateWithDefaults();
|
||||||
|
|
||||||
|
// Create the vision portal the easy way.
|
||||||
|
if (USE_WEBCAM) {
|
||||||
|
visionPortal = VisionPortal.easyCreateWithDefaults(
|
||||||
|
hardwareMap.get(WebcamName.class, "watch out"), tfod);
|
||||||
|
} else {
|
||||||
|
visionPortal = VisionPortal.easyCreateWithDefaults(
|
||||||
|
BuiltinCameraDirection.BACK, tfod);
|
||||||
|
}
|
||||||
|
|
||||||
|
} // end method initTfod()
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Add telemetry about TensorFlow Object Detection (TFOD) recognitions.
|
||||||
|
*/
|
||||||
|
private void telemetryTfod() {
|
||||||
|
|
||||||
|
List<Recognition> currentRecognitions = tfod.getRecognitions();
|
||||||
|
telemetry.addData("# Objects Detected", currentRecognitions.size());
|
||||||
|
|
||||||
|
// Step through the list of recognitions and display info for each one.
|
||||||
|
for (Recognition recognition : currentRecognitions) {
|
||||||
|
double x = (recognition.getLeft() + recognition.getRight()) / 2 ;
|
||||||
|
double y = (recognition.getTop() + recognition.getBottom()) / 2 ;
|
||||||
|
|
||||||
|
telemetry.addData(""," ");
|
||||||
|
telemetry.addData("Image", "%s (%.0f %% Conf.)", recognition.getLabel(), recognition.getConfidence() * 100);
|
||||||
|
telemetry.addData("- Position", "%.0f / %.0f", x, y);
|
||||||
|
telemetry.addData("- Size", "%.0f x %.0f", recognition.getWidth(), recognition.getHeight());
|
||||||
|
} // end for() loop
|
||||||
|
|
||||||
|
} // end method telemetryTfod()
|
||||||
|
|
||||||
|
} // end class
|
@ -0,0 +1,22 @@
|
|||||||
|
package org.firstinspires.ftc.teamcode;
|
||||||
|
|
||||||
|
import com.acmerobotics.dashboard.canvas.Canvas;
|
||||||
|
import com.acmerobotics.roadrunner.Pose2d;
|
||||||
|
import com.acmerobotics.roadrunner.Vector2d;
|
||||||
|
|
||||||
|
public final class Drawing {
|
||||||
|
private Drawing() {}
|
||||||
|
|
||||||
|
|
||||||
|
public static void drawRobot(Canvas c, Pose2d t) {
|
||||||
|
final double ROBOT_RADIUS = 9;
|
||||||
|
|
||||||
|
c.setStrokeWidth(1);
|
||||||
|
c.strokeCircle(t.position.x, t.position.y, ROBOT_RADIUS);
|
||||||
|
|
||||||
|
Vector2d halfv = t.heading.vec().times(0.5 * ROBOT_RADIUS);
|
||||||
|
Vector2d p1 = t.position.plus(halfv);
|
||||||
|
Vector2d p2 = p1.plus(halfv);
|
||||||
|
c.strokeLine(p1.x, p1.y, p2.x, p2.y);
|
||||||
|
}
|
||||||
|
}
|
@ -0,0 +1,8 @@
|
|||||||
|
package org.firstinspires.ftc.teamcode;
|
||||||
|
|
||||||
|
import com.acmerobotics.roadrunner.Time;
|
||||||
|
import com.acmerobotics.roadrunner.Twist2dDual;
|
||||||
|
|
||||||
|
public interface Localizer {
|
||||||
|
Twist2dDual<Time> update();
|
||||||
|
}
|
@ -0,0 +1,502 @@
|
|||||||
|
package org.firstinspires.ftc.teamcode;
|
||||||
|
|
||||||
|
import static org.firstinspires.ftc.teamcode.cometbots.Constants.*;
|
||||||
|
|
||||||
|
import androidx.annotation.NonNull;
|
||||||
|
|
||||||
|
import com.acmerobotics.dashboard.canvas.Canvas;
|
||||||
|
import com.acmerobotics.dashboard.config.Config;
|
||||||
|
import com.acmerobotics.dashboard.telemetry.TelemetryPacket;
|
||||||
|
import com.acmerobotics.roadrunner.*;
|
||||||
|
import com.acmerobotics.roadrunner.AngularVelConstraint;
|
||||||
|
import com.acmerobotics.roadrunner.DualNum;
|
||||||
|
import com.acmerobotics.roadrunner.HolonomicController;
|
||||||
|
import com.acmerobotics.roadrunner.MecanumKinematics;
|
||||||
|
import com.acmerobotics.roadrunner.MinVelConstraint;
|
||||||
|
import com.acmerobotics.roadrunner.MotorFeedforward;
|
||||||
|
import com.acmerobotics.roadrunner.Pose2d;
|
||||||
|
import com.acmerobotics.roadrunner.Pose2dDual;
|
||||||
|
import com.acmerobotics.roadrunner.ProfileAccelConstraint;
|
||||||
|
import com.acmerobotics.roadrunner.Time;
|
||||||
|
import com.acmerobotics.roadrunner.TimeTrajectory;
|
||||||
|
import com.acmerobotics.roadrunner.TimeTurn;
|
||||||
|
import com.acmerobotics.roadrunner.TrajectoryActionBuilder;
|
||||||
|
import com.acmerobotics.roadrunner.TurnConstraints;
|
||||||
|
import com.acmerobotics.roadrunner.Twist2dDual;
|
||||||
|
import com.acmerobotics.roadrunner.VelConstraint;
|
||||||
|
import com.acmerobotics.roadrunner.ftc.DownsampledWriter;
|
||||||
|
import com.acmerobotics.roadrunner.ftc.Encoder;
|
||||||
|
import com.acmerobotics.roadrunner.ftc.FlightRecorder;
|
||||||
|
import com.acmerobotics.roadrunner.ftc.LazyImu;
|
||||||
|
import com.acmerobotics.roadrunner.ftc.LynxFirmware;
|
||||||
|
import com.acmerobotics.roadrunner.ftc.OverflowEncoder;
|
||||||
|
import com.acmerobotics.roadrunner.ftc.PositionVelocityPair;
|
||||||
|
import com.acmerobotics.roadrunner.ftc.RawEncoder;
|
||||||
|
import com.qualcomm.hardware.lynx.LynxModule;
|
||||||
|
import com.qualcomm.hardware.rev.RevHubOrientationOnRobot;
|
||||||
|
import com.qualcomm.robotcore.hardware.DcMotor;
|
||||||
|
import com.qualcomm.robotcore.hardware.DcMotorEx;
|
||||||
|
import com.qualcomm.robotcore.hardware.DcMotorSimple;
|
||||||
|
import com.qualcomm.robotcore.hardware.HardwareMap;
|
||||||
|
import com.qualcomm.robotcore.hardware.IMU;
|
||||||
|
import com.qualcomm.robotcore.hardware.VoltageSensor;
|
||||||
|
|
||||||
|
import org.firstinspires.ftc.robotcore.external.navigation.AngleUnit;
|
||||||
|
import org.firstinspires.ftc.robotcore.external.navigation.YawPitchRollAngles;
|
||||||
|
import org.firstinspires.ftc.teamcode.cometbots.MovementLibrary;
|
||||||
|
import org.firstinspires.ftc.teamcode.messages.DriveCommandMessage;
|
||||||
|
import org.firstinspires.ftc.teamcode.messages.MecanumCommandMessage;
|
||||||
|
import org.firstinspires.ftc.teamcode.messages.MecanumLocalizerInputsMessage;
|
||||||
|
import org.firstinspires.ftc.teamcode.messages.PoseMessage;
|
||||||
|
|
||||||
|
import java.lang.Math;
|
||||||
|
import java.util.Arrays;
|
||||||
|
import java.util.LinkedList;
|
||||||
|
import java.util.List;
|
||||||
|
|
||||||
|
@Config
|
||||||
|
public final class MecanumDrive {
|
||||||
|
public static class Params {
|
||||||
|
// IMU orientation
|
||||||
|
// TODO: fill in these values based on
|
||||||
|
// see https://ftc-docs.firstinspires.org/en/latest/programming_resources/imu/imu.html?highlight=imu#physical-hub-mounting
|
||||||
|
public RevHubOrientationOnRobot.LogoFacingDirection logoFacingDirection =
|
||||||
|
RevHubOrientationOnRobot.LogoFacingDirection.FORWARD;
|
||||||
|
public RevHubOrientationOnRobot.UsbFacingDirection usbFacingDirection =
|
||||||
|
RevHubOrientationOnRobot.UsbFacingDirection.UP;
|
||||||
|
|
||||||
|
// drive model parameters
|
||||||
|
public double inPerTick = 119/5924.5;
|
||||||
|
public double lateralInPerTick = 124/6930.5;
|
||||||
|
public double trackWidthTicks = 1131;
|
||||||
|
|
||||||
|
// feedforward parameters (in tick units)
|
||||||
|
public double kS = 0.7476;
|
||||||
|
public double kV = 0.0043736;
|
||||||
|
public double kA = 00055;
|
||||||
|
|
||||||
|
// path profile parameters (in inches)
|
||||||
|
public double maxWheelVel = 50;
|
||||||
|
public double minProfileAccel = -30;
|
||||||
|
public double maxProfileAccel = 50;
|
||||||
|
|
||||||
|
// turn profile parameters (in radians)
|
||||||
|
public double maxAngVel = Math.PI; // shared with path
|
||||||
|
public double maxAngAccel = Math.PI;
|
||||||
|
|
||||||
|
// path controller gains
|
||||||
|
public double axialGain = 4.125;
|
||||||
|
public double lateralGain = 4.125;
|
||||||
|
public double headingGain = 4.25; // shared with turn
|
||||||
|
|
||||||
|
public double axialVelGain = 0.0;
|
||||||
|
public double lateralVelGain = 0.0;
|
||||||
|
public double headingVelGain = 0.0; // shared with turn
|
||||||
|
}
|
||||||
|
|
||||||
|
public static Params PARAMS = new Params();
|
||||||
|
|
||||||
|
public final MecanumKinematics kinematics = new MecanumKinematics(
|
||||||
|
PARAMS.inPerTick * PARAMS.trackWidthTicks, PARAMS.inPerTick / PARAMS.lateralInPerTick);
|
||||||
|
|
||||||
|
public final TurnConstraints defaultTurnConstraints = new TurnConstraints(
|
||||||
|
PARAMS.maxAngVel, -PARAMS.maxAngAccel, PARAMS.maxAngAccel);
|
||||||
|
public final VelConstraint defaultVelConstraint =
|
||||||
|
new MinVelConstraint(Arrays.asList(
|
||||||
|
kinematics.new WheelVelConstraint(PARAMS.maxWheelVel),
|
||||||
|
new AngularVelConstraint(PARAMS.maxAngVel)
|
||||||
|
));
|
||||||
|
public final AccelConstraint defaultAccelConstraint =
|
||||||
|
new ProfileAccelConstraint(PARAMS.minProfileAccel, PARAMS.maxProfileAccel);
|
||||||
|
|
||||||
|
public final DcMotorEx leftFront, leftBack, rightBack, rightFront;
|
||||||
|
|
||||||
|
public final VoltageSensor voltageSensor;
|
||||||
|
|
||||||
|
public final LazyImu lazyImu;
|
||||||
|
|
||||||
|
public final Localizer localizer;
|
||||||
|
public Pose2d pose;
|
||||||
|
|
||||||
|
private final LinkedList<Pose2d> poseHistory = new LinkedList<>();
|
||||||
|
|
||||||
|
private final DownsampledWriter estimatedPoseWriter = new DownsampledWriter("ESTIMATED_POSE", 50_000_000);
|
||||||
|
private final DownsampledWriter targetPoseWriter = new DownsampledWriter("TARGET_POSE", 50_000_000);
|
||||||
|
private final DownsampledWriter driveCommandWriter = new DownsampledWriter("DRIVE_COMMAND", 50_000_000);
|
||||||
|
private final DownsampledWriter mecanumCommandWriter = new DownsampledWriter("MECANUM_COMMAND", 50_000_000);
|
||||||
|
|
||||||
|
public class DriveLocalizer implements Localizer {
|
||||||
|
public final Encoder leftFront, leftBack, rightBack, rightFront;
|
||||||
|
public final IMU imu;
|
||||||
|
|
||||||
|
private int lastLeftFrontPos, lastLeftBackPos, lastRightBackPos, lastRightFrontPos;
|
||||||
|
private Rotation2d lastHeading;
|
||||||
|
private boolean initialized;
|
||||||
|
|
||||||
|
public DriveLocalizer() {
|
||||||
|
leftFront = new OverflowEncoder(new RawEncoder(MecanumDrive.this.leftFront));
|
||||||
|
leftBack = new OverflowEncoder(new RawEncoder(MecanumDrive.this.leftBack));
|
||||||
|
rightBack = new OverflowEncoder(new RawEncoder(MecanumDrive.this.rightBack));
|
||||||
|
rightFront = new OverflowEncoder(new RawEncoder(MecanumDrive.this.rightFront));
|
||||||
|
|
||||||
|
imu = lazyImu.get();
|
||||||
|
|
||||||
|
// TODO: reverse encoders if needed
|
||||||
|
// leftFront.setDirection(DcMotorSimple.Direction.REVERSE);
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public Twist2dDual<Time> update() {
|
||||||
|
PositionVelocityPair leftFrontPosVel = leftFront.getPositionAndVelocity();
|
||||||
|
PositionVelocityPair leftBackPosVel = leftBack.getPositionAndVelocity();
|
||||||
|
PositionVelocityPair rightBackPosVel = rightBack.getPositionAndVelocity();
|
||||||
|
PositionVelocityPair rightFrontPosVel = rightFront.getPositionAndVelocity();
|
||||||
|
|
||||||
|
YawPitchRollAngles angles = imu.getRobotYawPitchRollAngles();
|
||||||
|
|
||||||
|
FlightRecorder.write("MECANUM_LOCALIZER_INPUTS", new MecanumLocalizerInputsMessage(
|
||||||
|
leftFrontPosVel, leftBackPosVel, rightBackPosVel, rightFrontPosVel, angles));
|
||||||
|
|
||||||
|
Rotation2d heading = Rotation2d.exp(angles.getYaw(AngleUnit.RADIANS));
|
||||||
|
|
||||||
|
if (!initialized) {
|
||||||
|
initialized = true;
|
||||||
|
|
||||||
|
lastLeftFrontPos = leftFrontPosVel.position;
|
||||||
|
lastLeftBackPos = leftBackPosVel.position;
|
||||||
|
lastRightBackPos = rightBackPosVel.position;
|
||||||
|
lastRightFrontPos = rightFrontPosVel.position;
|
||||||
|
|
||||||
|
lastHeading = heading;
|
||||||
|
|
||||||
|
return new Twist2dDual<>(
|
||||||
|
Vector2dDual.constant(new Vector2d(0.0, 0.0), 2),
|
||||||
|
DualNum.constant(0.0, 2)
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
double headingDelta = heading.minus(lastHeading);
|
||||||
|
Twist2dDual<Time> twist = kinematics.forward(new MecanumKinematics.WheelIncrements<>(
|
||||||
|
new DualNum<Time>(new double[]{
|
||||||
|
(leftFrontPosVel.position - lastLeftFrontPos),
|
||||||
|
leftFrontPosVel.velocity,
|
||||||
|
}).times(PARAMS.inPerTick),
|
||||||
|
new DualNum<Time>(new double[]{
|
||||||
|
(leftBackPosVel.position - lastLeftBackPos),
|
||||||
|
leftBackPosVel.velocity,
|
||||||
|
}).times(PARAMS.inPerTick),
|
||||||
|
new DualNum<Time>(new double[]{
|
||||||
|
(rightBackPosVel.position - lastRightBackPos),
|
||||||
|
rightBackPosVel.velocity,
|
||||||
|
}).times(PARAMS.inPerTick),
|
||||||
|
new DualNum<Time>(new double[]{
|
||||||
|
(rightFrontPosVel.position - lastRightFrontPos),
|
||||||
|
rightFrontPosVel.velocity,
|
||||||
|
}).times(PARAMS.inPerTick)
|
||||||
|
));
|
||||||
|
|
||||||
|
lastLeftFrontPos = leftFrontPosVel.position;
|
||||||
|
lastLeftBackPos = leftBackPosVel.position;
|
||||||
|
lastRightBackPos = rightBackPosVel.position;
|
||||||
|
lastRightFrontPos = rightFrontPosVel.position;
|
||||||
|
|
||||||
|
lastHeading = heading;
|
||||||
|
|
||||||
|
return new Twist2dDual<>(
|
||||||
|
twist.line,
|
||||||
|
DualNum.cons(headingDelta, twist.angle.drop(1))
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public MecanumDrive(HardwareMap hardwareMap, Pose2d pose) {
|
||||||
|
this.pose = pose;
|
||||||
|
|
||||||
|
LynxFirmware.throwIfModulesAreOutdated(hardwareMap);
|
||||||
|
|
||||||
|
for (LynxModule module : hardwareMap.getAll(LynxModule.class)) {
|
||||||
|
module.setBulkCachingMode(LynxModule.BulkCachingMode.AUTO);
|
||||||
|
}
|
||||||
|
|
||||||
|
// TODO: make sure your config has motors with these names (or change them)
|
||||||
|
// see https://ftc-docs.firstinspires.org/en/latest/hardware_and_software_configuration/configuring/index.html
|
||||||
|
leftFront = hardwareMap.get(DcMotorEx.class, LT_FRONT);
|
||||||
|
leftBack = hardwareMap.get(DcMotorEx.class, LT_BACK);
|
||||||
|
rightBack = hardwareMap.get(DcMotorEx.class, RT_BACK);
|
||||||
|
rightFront = hardwareMap.get(DcMotorEx.class, RT_FRONT);
|
||||||
|
|
||||||
|
leftFront.setZeroPowerBehavior(DcMotor.ZeroPowerBehavior.BRAKE);
|
||||||
|
leftBack.setZeroPowerBehavior(DcMotor.ZeroPowerBehavior.BRAKE);
|
||||||
|
rightBack.setZeroPowerBehavior(DcMotor.ZeroPowerBehavior.BRAKE);
|
||||||
|
rightFront.setZeroPowerBehavior(DcMotor.ZeroPowerBehavior.BRAKE);
|
||||||
|
|
||||||
|
/*
|
||||||
|
Put all motors in an array for easier handling
|
||||||
|
|
||||||
|
NOTE: Placement of motor in array is important especially when dealing with functions
|
||||||
|
that set direction
|
||||||
|
*/
|
||||||
|
DcMotor[] motors = {rightFront, rightBack, leftFront, leftBack};
|
||||||
|
|
||||||
|
/*
|
||||||
|
Initialize all wheels forward using motors a
|
||||||
|
*/
|
||||||
|
MovementLibrary.setDirectionForward(motors);
|
||||||
|
|
||||||
|
// TODO: make sure your config has an IMU with this name (can be BNO or BHI)
|
||||||
|
// see https://ftc-docs.firstinspires.org/en/latest/hardware_and_software_configuration/configuring/index.html
|
||||||
|
lazyImu = new LazyImu(hardwareMap, "imu", new RevHubOrientationOnRobot(
|
||||||
|
PARAMS.logoFacingDirection, PARAMS.usbFacingDirection));
|
||||||
|
|
||||||
|
voltageSensor = hardwareMap.voltageSensor.iterator().next();
|
||||||
|
|
||||||
|
localizer = new DriveLocalizer();
|
||||||
|
|
||||||
|
FlightRecorder.write("MECANUM_PARAMS", PARAMS);
|
||||||
|
}
|
||||||
|
|
||||||
|
public void setDrivePowers(PoseVelocity2d powers) {
|
||||||
|
MecanumKinematics.WheelVelocities<Time> wheelVels = new MecanumKinematics(1).inverse(
|
||||||
|
PoseVelocity2dDual.constant(powers, 1));
|
||||||
|
|
||||||
|
double maxPowerMag = 1;
|
||||||
|
for (DualNum<Time> power : wheelVels.all()) {
|
||||||
|
maxPowerMag = Math.max(maxPowerMag, power.value());
|
||||||
|
}
|
||||||
|
|
||||||
|
leftFront.setPower(wheelVels.leftFront.get(0) / maxPowerMag);
|
||||||
|
leftBack.setPower(wheelVels.leftBack.get(0) / maxPowerMag);
|
||||||
|
rightBack.setPower(wheelVels.rightBack.get(0) / maxPowerMag);
|
||||||
|
rightFront.setPower(wheelVels.rightFront.get(0) / maxPowerMag);
|
||||||
|
}
|
||||||
|
|
||||||
|
public final class FollowTrajectoryAction implements Action {
|
||||||
|
public final TimeTrajectory timeTrajectory;
|
||||||
|
private double beginTs = -1;
|
||||||
|
|
||||||
|
private final double[] xPoints, yPoints;
|
||||||
|
|
||||||
|
public FollowTrajectoryAction(TimeTrajectory t) {
|
||||||
|
timeTrajectory = t;
|
||||||
|
|
||||||
|
List<Double> disps = com.acmerobotics.roadrunner.Math.range(
|
||||||
|
0, t.path.length(),
|
||||||
|
Math.max(2, (int) Math.ceil(t.path.length() / 2)));
|
||||||
|
xPoints = new double[disps.size()];
|
||||||
|
yPoints = new double[disps.size()];
|
||||||
|
for (int i = 0; i < disps.size(); i++) {
|
||||||
|
Pose2d p = t.path.get(disps.get(i), 1).value();
|
||||||
|
xPoints[i] = p.position.x;
|
||||||
|
yPoints[i] = p.position.y;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public boolean run(@NonNull TelemetryPacket p) {
|
||||||
|
double t;
|
||||||
|
if (beginTs < 0) {
|
||||||
|
beginTs = Actions.now();
|
||||||
|
t = 0;
|
||||||
|
} else {
|
||||||
|
t = Actions.now() - beginTs;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (t >= timeTrajectory.duration) {
|
||||||
|
leftFront.setPower(0);
|
||||||
|
leftBack.setPower(0);
|
||||||
|
rightBack.setPower(0);
|
||||||
|
rightFront.setPower(0);
|
||||||
|
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
Pose2dDual<Time> txWorldTarget = timeTrajectory.get(t);
|
||||||
|
targetPoseWriter.write(new PoseMessage(txWorldTarget.value()));
|
||||||
|
|
||||||
|
PoseVelocity2d robotVelRobot = updatePoseEstimate();
|
||||||
|
|
||||||
|
PoseVelocity2dDual<Time> command = new HolonomicController(
|
||||||
|
PARAMS.axialGain, PARAMS.lateralGain, PARAMS.headingGain,
|
||||||
|
PARAMS.axialVelGain, PARAMS.lateralVelGain, PARAMS.headingVelGain
|
||||||
|
)
|
||||||
|
.compute(txWorldTarget, pose, robotVelRobot);
|
||||||
|
driveCommandWriter.write(new DriveCommandMessage(command));
|
||||||
|
|
||||||
|
MecanumKinematics.WheelVelocities<Time> wheelVels = kinematics.inverse(command);
|
||||||
|
double voltage = voltageSensor.getVoltage();
|
||||||
|
|
||||||
|
final MotorFeedforward feedforward = new MotorFeedforward(PARAMS.kS,
|
||||||
|
PARAMS.kV / PARAMS.inPerTick, PARAMS.kA / PARAMS.inPerTick);
|
||||||
|
double leftFrontPower = feedforward.compute(wheelVels.leftFront) / voltage;
|
||||||
|
double leftBackPower = feedforward.compute(wheelVels.leftBack) / voltage;
|
||||||
|
double rightBackPower = feedforward.compute(wheelVels.rightBack) / voltage;
|
||||||
|
double rightFrontPower = feedforward.compute(wheelVels.rightFront) / voltage;
|
||||||
|
mecanumCommandWriter.write(new MecanumCommandMessage(
|
||||||
|
voltage, leftFrontPower, leftBackPower, rightBackPower, rightFrontPower
|
||||||
|
));
|
||||||
|
|
||||||
|
leftFront.setPower(leftFrontPower);
|
||||||
|
leftBack.setPower(leftBackPower);
|
||||||
|
rightBack.setPower(rightBackPower);
|
||||||
|
rightFront.setPower(rightFrontPower);
|
||||||
|
|
||||||
|
p.put("x", pose.position.x);
|
||||||
|
p.put("y", pose.position.y);
|
||||||
|
p.put("heading (deg)", Math.toDegrees(pose.heading.toDouble()));
|
||||||
|
|
||||||
|
Pose2d error = txWorldTarget.value().minusExp(pose);
|
||||||
|
p.put("xError", error.position.x);
|
||||||
|
p.put("yError", error.position.y);
|
||||||
|
p.put("headingError (deg)", Math.toDegrees(error.heading.toDouble()));
|
||||||
|
|
||||||
|
// only draw when active; only one drive action should be active at a time
|
||||||
|
Canvas c = p.fieldOverlay();
|
||||||
|
drawPoseHistory(c);
|
||||||
|
|
||||||
|
c.setStroke("#4CAF50");
|
||||||
|
Drawing.drawRobot(c, txWorldTarget.value());
|
||||||
|
|
||||||
|
c.setStroke("#3F51B5");
|
||||||
|
Drawing.drawRobot(c, pose);
|
||||||
|
|
||||||
|
c.setStroke("#4CAF50FF");
|
||||||
|
c.setStrokeWidth(1);
|
||||||
|
c.strokePolyline(xPoints, yPoints);
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void preview(Canvas c) {
|
||||||
|
c.setStroke("#4CAF507A");
|
||||||
|
c.setStrokeWidth(1);
|
||||||
|
c.strokePolyline(xPoints, yPoints);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public final class TurnAction implements Action {
|
||||||
|
private final TimeTurn turn;
|
||||||
|
|
||||||
|
private double beginTs = -1;
|
||||||
|
|
||||||
|
public TurnAction(TimeTurn turn) {
|
||||||
|
this.turn = turn;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public boolean run(@NonNull TelemetryPacket p) {
|
||||||
|
double t;
|
||||||
|
if (beginTs < 0) {
|
||||||
|
beginTs = Actions.now();
|
||||||
|
t = 0;
|
||||||
|
} else {
|
||||||
|
t = Actions.now() - beginTs;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (t >= turn.duration) {
|
||||||
|
leftFront.setPower(0);
|
||||||
|
leftBack.setPower(0);
|
||||||
|
rightBack.setPower(0);
|
||||||
|
rightFront.setPower(0);
|
||||||
|
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
Pose2dDual<Time> txWorldTarget = turn.get(t);
|
||||||
|
targetPoseWriter.write(new PoseMessage(txWorldTarget.value()));
|
||||||
|
|
||||||
|
PoseVelocity2d robotVelRobot = updatePoseEstimate();
|
||||||
|
|
||||||
|
PoseVelocity2dDual<Time> command = new HolonomicController(
|
||||||
|
PARAMS.axialGain, PARAMS.lateralGain, PARAMS.headingGain,
|
||||||
|
PARAMS.axialVelGain, PARAMS.lateralVelGain, PARAMS.headingVelGain
|
||||||
|
)
|
||||||
|
.compute(txWorldTarget, pose, robotVelRobot);
|
||||||
|
driveCommandWriter.write(new DriveCommandMessage(command));
|
||||||
|
|
||||||
|
MecanumKinematics.WheelVelocities<Time> wheelVels = kinematics.inverse(command);
|
||||||
|
double voltage = voltageSensor.getVoltage();
|
||||||
|
final MotorFeedforward feedforward = new MotorFeedforward(PARAMS.kS,
|
||||||
|
PARAMS.kV / PARAMS.inPerTick, PARAMS.kA / PARAMS.inPerTick);
|
||||||
|
double leftFrontPower = feedforward.compute(wheelVels.leftFront) / voltage;
|
||||||
|
double leftBackPower = feedforward.compute(wheelVels.leftBack) / voltage;
|
||||||
|
double rightBackPower = feedforward.compute(wheelVels.rightBack) / voltage;
|
||||||
|
double rightFrontPower = feedforward.compute(wheelVels.rightFront) / voltage;
|
||||||
|
mecanumCommandWriter.write(new MecanumCommandMessage(
|
||||||
|
voltage, leftFrontPower, leftBackPower, rightBackPower, rightFrontPower
|
||||||
|
));
|
||||||
|
|
||||||
|
leftFront.setPower(feedforward.compute(wheelVels.leftFront) / voltage);
|
||||||
|
leftBack.setPower(feedforward.compute(wheelVels.leftBack) / voltage);
|
||||||
|
rightBack.setPower(feedforward.compute(wheelVels.rightBack) / voltage);
|
||||||
|
rightFront.setPower(feedforward.compute(wheelVels.rightFront) / voltage);
|
||||||
|
|
||||||
|
Canvas c = p.fieldOverlay();
|
||||||
|
drawPoseHistory(c);
|
||||||
|
|
||||||
|
c.setStroke("#4CAF50");
|
||||||
|
Drawing.drawRobot(c, txWorldTarget.value());
|
||||||
|
|
||||||
|
c.setStroke("#3F51B5");
|
||||||
|
Drawing.drawRobot(c, pose);
|
||||||
|
|
||||||
|
c.setStroke("#7C4DFFFF");
|
||||||
|
c.fillCircle(turn.beginPose.position.x, turn.beginPose.position.y, 2);
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void preview(Canvas c) {
|
||||||
|
c.setStroke("#7C4DFF7A");
|
||||||
|
c.fillCircle(turn.beginPose.position.x, turn.beginPose.position.y, 2);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public PoseVelocity2d updatePoseEstimate() {
|
||||||
|
Twist2dDual<Time> twist = localizer.update();
|
||||||
|
pose = pose.plus(twist.value());
|
||||||
|
|
||||||
|
poseHistory.add(pose);
|
||||||
|
while (poseHistory.size() > 100) {
|
||||||
|
poseHistory.removeFirst();
|
||||||
|
}
|
||||||
|
|
||||||
|
estimatedPoseWriter.write(new PoseMessage(pose));
|
||||||
|
|
||||||
|
return twist.velocity().value();
|
||||||
|
}
|
||||||
|
|
||||||
|
private void drawPoseHistory(Canvas c) {
|
||||||
|
double[] xPoints = new double[poseHistory.size()];
|
||||||
|
double[] yPoints = new double[poseHistory.size()];
|
||||||
|
|
||||||
|
int i = 0;
|
||||||
|
for (Pose2d t : poseHistory) {
|
||||||
|
xPoints[i] = t.position.x;
|
||||||
|
yPoints[i] = t.position.y;
|
||||||
|
|
||||||
|
i++;
|
||||||
|
}
|
||||||
|
|
||||||
|
c.setStrokeWidth(1);
|
||||||
|
c.setStroke("#3F51B5");
|
||||||
|
c.strokePolyline(xPoints, yPoints);
|
||||||
|
}
|
||||||
|
|
||||||
|
public TrajectoryActionBuilder actionBuilder(Pose2d beginPose) {
|
||||||
|
return new TrajectoryActionBuilder(
|
||||||
|
TurnAction::new,
|
||||||
|
FollowTrajectoryAction::new,
|
||||||
|
new TrajectoryBuilderParams(
|
||||||
|
1e-6,
|
||||||
|
new ProfileParams(
|
||||||
|
0.25, 0.1, 1e-2
|
||||||
|
)
|
||||||
|
),
|
||||||
|
beginPose, 0.0,
|
||||||
|
defaultTurnConstraints,
|
||||||
|
defaultVelConstraint, defaultAccelConstraint
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
@ -0,0 +1,12 @@
|
|||||||
|
package org.firstinspires.ftc.teamcode.cometbots;
|
||||||
|
|
||||||
|
public class Constants {
|
||||||
|
|
||||||
|
public static double CB_BASE_SPEED = 0.2;
|
||||||
|
|
||||||
|
public static String RT_FRONT = "right hand wheel";
|
||||||
|
public static String LT_FRONT = "left hand wheel";
|
||||||
|
public static String RT_BACK = "right leg wheel";
|
||||||
|
public static String LT_BACK = "left leg wheel";
|
||||||
|
|
||||||
|
}
|
@ -0,0 +1,58 @@
|
|||||||
|
package org.firstinspires.ftc.teamcode.cometbots;
|
||||||
|
|
||||||
|
import com.qualcomm.robotcore.hardware.DcMotor;
|
||||||
|
import com.qualcomm.robotcore.hardware.DcMotorSimple;
|
||||||
|
|
||||||
|
public class MovementLibrary {
|
||||||
|
|
||||||
|
public static void setDirectionForward(DcMotor[] motors) {
|
||||||
|
setLeftSideForward(motors);
|
||||||
|
setRightSideForward(motors);
|
||||||
|
}
|
||||||
|
|
||||||
|
public static void setLeftSideForward(DcMotor[] motors) {
|
||||||
|
motors[2].setDirection(DcMotor.Direction.FORWARD);
|
||||||
|
motors[3].setDirection(DcMotor.Direction.FORWARD);
|
||||||
|
}
|
||||||
|
|
||||||
|
public static void setRightSideForward(DcMotor[] motors) {
|
||||||
|
motors[0].setDirection(DcMotor.Direction.FORWARD);
|
||||||
|
motors[1].setDirection(DcMotor.Direction.FORWARD);
|
||||||
|
}
|
||||||
|
|
||||||
|
public static void setLeftSideBackward(DcMotor[] motors) {
|
||||||
|
motors[2].setDirection(DcMotor.Direction.REVERSE);
|
||||||
|
motors[3].setDirection(DcMotor.Direction.REVERSE);
|
||||||
|
}
|
||||||
|
|
||||||
|
public static void setRightSideBackward(DcMotor[] motors) {
|
||||||
|
motors[0].setDirection(DcMotor.Direction.REVERSE);
|
||||||
|
motors[1].setDirection(DcMotor.Direction.REVERSE);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
public static void setDirectionBackward(DcMotor[] motors) {
|
||||||
|
setLeftSideBackward(motors);
|
||||||
|
setRightSideBackward(motors);
|
||||||
|
}
|
||||||
|
|
||||||
|
public static void setSpeed(DcMotor[] motors, double speedValue) {
|
||||||
|
for (DcMotor motor : motors) {
|
||||||
|
motor.setPower(speedValue);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public static void setDirectionLeft(DcMotor[] motors) {
|
||||||
|
motors[0].setDirection(DcMotor.Direction.REVERSE);
|
||||||
|
motors[1].setDirection(DcMotor.Direction.REVERSE);
|
||||||
|
motors[2].setDirection(DcMotor.Direction.FORWARD);
|
||||||
|
motors[3].setDirection(DcMotor.Direction.FORWARD);
|
||||||
|
}
|
||||||
|
|
||||||
|
public static void setDirectionRight(DcMotor[] motors) {
|
||||||
|
motors[0].setDirection(DcMotor.Direction.FORWARD);
|
||||||
|
motors[1].setDirection(DcMotor.Direction.FORWARD);
|
||||||
|
motors[2].setDirection(DcMotor.Direction.REVERSE);
|
||||||
|
motors[3].setDirection(DcMotor.Direction.REVERSE);
|
||||||
|
}
|
||||||
|
}
|
@ -0,0 +1,24 @@
|
|||||||
|
package org.firstinspires.ftc.teamcode.messages;
|
||||||
|
|
||||||
|
import com.acmerobotics.roadrunner.PoseVelocity2dDual;
|
||||||
|
import com.acmerobotics.roadrunner.Time;
|
||||||
|
|
||||||
|
public final class DriveCommandMessage {
|
||||||
|
public long timestamp;
|
||||||
|
public double forwardVelocity;
|
||||||
|
public double forwardAcceleration;
|
||||||
|
public double lateralVelocity;
|
||||||
|
public double lateralAcceleration;
|
||||||
|
public double angularVelocity;
|
||||||
|
public double angularAcceleration;
|
||||||
|
|
||||||
|
public DriveCommandMessage(PoseVelocity2dDual<Time> poseVelocity) {
|
||||||
|
this.timestamp = System.nanoTime();
|
||||||
|
this.forwardVelocity = poseVelocity.linearVel.x.get(0);
|
||||||
|
this.forwardAcceleration = poseVelocity.linearVel.x.get(1);
|
||||||
|
this.lateralVelocity = poseVelocity.linearVel.y.get(0);
|
||||||
|
this.lateralAcceleration = poseVelocity.linearVel.y.get(1);
|
||||||
|
this.angularVelocity = poseVelocity.angVel.get(0);
|
||||||
|
this.angularAcceleration = poseVelocity.angVel.get(1);
|
||||||
|
}
|
||||||
|
}
|
@ -0,0 +1,19 @@
|
|||||||
|
package org.firstinspires.ftc.teamcode.messages;
|
||||||
|
|
||||||
|
public final class MecanumCommandMessage {
|
||||||
|
public long timestamp;
|
||||||
|
public double voltage;
|
||||||
|
public double leftFrontPower;
|
||||||
|
public double leftBackPower;
|
||||||
|
public double rightBackPower;
|
||||||
|
public double rightFrontPower;
|
||||||
|
|
||||||
|
public MecanumCommandMessage(double voltage, double leftFrontPower, double leftBackPower, double rightBackPower, double rightFrontPower) {
|
||||||
|
this.timestamp = System.nanoTime();
|
||||||
|
this.voltage = voltage;
|
||||||
|
this.leftFrontPower = leftFrontPower;
|
||||||
|
this.leftBackPower = leftBackPower;
|
||||||
|
this.rightBackPower = rightBackPower;
|
||||||
|
this.rightFrontPower = rightFrontPower;
|
||||||
|
}
|
||||||
|
}
|
@ -0,0 +1,30 @@
|
|||||||
|
package org.firstinspires.ftc.teamcode.messages;
|
||||||
|
|
||||||
|
import com.acmerobotics.roadrunner.ftc.PositionVelocityPair;
|
||||||
|
|
||||||
|
import org.firstinspires.ftc.robotcore.external.navigation.AngleUnit;
|
||||||
|
import org.firstinspires.ftc.robotcore.external.navigation.YawPitchRollAngles;
|
||||||
|
|
||||||
|
public final class MecanumLocalizerInputsMessage {
|
||||||
|
public long timestamp;
|
||||||
|
public PositionVelocityPair leftFront;
|
||||||
|
public PositionVelocityPair leftBack;
|
||||||
|
public PositionVelocityPair rightBack;
|
||||||
|
public PositionVelocityPair rightFront;
|
||||||
|
public double yaw;
|
||||||
|
public double pitch;
|
||||||
|
public double roll;
|
||||||
|
|
||||||
|
public MecanumLocalizerInputsMessage(PositionVelocityPair leftFront, PositionVelocityPair leftBack, PositionVelocityPair rightBack, PositionVelocityPair rightFront, YawPitchRollAngles angles) {
|
||||||
|
this.timestamp = System.nanoTime();
|
||||||
|
this.leftFront = leftFront;
|
||||||
|
this.leftBack = leftBack;
|
||||||
|
this.rightBack = rightBack;
|
||||||
|
this.rightFront = rightFront;
|
||||||
|
{
|
||||||
|
this.yaw = angles.getYaw(AngleUnit.RADIANS);
|
||||||
|
this.pitch = angles.getPitch(AngleUnit.RADIANS);
|
||||||
|
this.roll = angles.getRoll(AngleUnit.RADIANS);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
@ -0,0 +1,17 @@
|
|||||||
|
package org.firstinspires.ftc.teamcode.messages;
|
||||||
|
|
||||||
|
import com.acmerobotics.roadrunner.Pose2d;
|
||||||
|
|
||||||
|
public final class PoseMessage {
|
||||||
|
public long timestamp;
|
||||||
|
public double x;
|
||||||
|
public double y;
|
||||||
|
public double heading;
|
||||||
|
|
||||||
|
public PoseMessage(Pose2d pose) {
|
||||||
|
this.timestamp = System.nanoTime();
|
||||||
|
this.x = pose.position.x;
|
||||||
|
this.y = pose.position.y;
|
||||||
|
this.heading = pose.heading.toDouble();
|
||||||
|
}
|
||||||
|
}
|
@ -0,0 +1,4 @@
|
|||||||
|
https://www.youtube.com/watch?v=aeMWWvteF2U
|
||||||
|
|
||||||
|
https://www.youtube.com/watch?v=1xdo8QP6_mI
|
||||||
|
|
@ -1,21 +1,23 @@
|
|||||||
repositories {
|
repositories {
|
||||||
mavenCentral()
|
mavenCentral()
|
||||||
google() // Needed for androidx
|
google() // Needed for androidx
|
||||||
|
maven { url = 'https://maven.brott.dev/' }
|
||||||
}
|
}
|
||||||
|
|
||||||
dependencies {
|
dependencies {
|
||||||
implementation 'org.firstinspires.ftc:Inspection:9.0.0'
|
implementation 'org.firstinspires.ftc:Inspection:9.1.0'
|
||||||
implementation 'org.firstinspires.ftc:Blocks:9.0.0'
|
implementation 'org.firstinspires.ftc:Blocks:9.1.0'
|
||||||
implementation 'org.firstinspires.ftc:Tfod:9.0.0'
|
implementation 'org.firstinspires.ftc:Tfod:9.1.0'
|
||||||
implementation 'org.firstinspires.ftc:RobotCore:9.0.0'
|
implementation 'org.firstinspires.ftc:RobotCore:9.1.0'
|
||||||
implementation 'org.firstinspires.ftc:RobotServer:9.0.0'
|
implementation 'org.firstinspires.ftc:RobotServer:9.1.0'
|
||||||
implementation 'org.firstinspires.ftc:OnBotJava:9.0.0'
|
implementation 'org.firstinspires.ftc:OnBotJava:9.1.0'
|
||||||
implementation 'org.firstinspires.ftc:Hardware:9.0.0'
|
implementation 'org.firstinspires.ftc:Hardware:9.1.0'
|
||||||
implementation 'org.firstinspires.ftc:FtcCommon:9.0.0'
|
implementation 'org.firstinspires.ftc:FtcCommon:9.1.0'
|
||||||
implementation 'org.firstinspires.ftc:Vision:9.0.0'
|
implementation 'org.firstinspires.ftc:Vision:9.1.0'
|
||||||
implementation 'org.firstinspires.ftc:gameAssets-CenterStage:1.0.0'
|
implementation 'org.firstinspires.ftc:gameAssets-CenterStage:1.0.0'
|
||||||
implementation 'org.tensorflow:tensorflow-lite-task-vision:0.4.3'
|
implementation 'org.tensorflow:tensorflow-lite-task-vision:0.4.3'
|
||||||
runtimeOnly 'org.tensorflow:tensorflow-lite:2.12.0'
|
runtimeOnly 'org.tensorflow:tensorflow-lite:2.12.0'
|
||||||
implementation 'androidx.appcompat:appcompat:1.2.0'
|
implementation 'androidx.appcompat:appcompat:1.2.0'
|
||||||
|
implementation 'com.acmerobotics.dashboard:dashboard:0.4.15' // FTC Dashboard
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Reference in New Issue
Block a user