diff --git a/lib/widgets/draggable_containers/draggable_nt4_widget_container.dart b/lib/widgets/draggable_containers/draggable_nt4_widget_container.dart index 6af9419d..63dd38fd 100644 --- a/lib/widgets/draggable_containers/draggable_nt4_widget_container.dart +++ b/lib/widgets/draggable_containers/draggable_nt4_widget_container.dart @@ -14,6 +14,7 @@ import 'package:elastic_dashboard/widgets/nt4_widgets/multi-topic/combo_box_choo import 'package:elastic_dashboard/widgets/nt4_widgets/multi-topic/robot_preferences.dart'; import 'package:elastic_dashboard/widgets/nt4_widgets/multi-topic/split_button_chooser.dart'; import 'package:elastic_dashboard/widgets/nt4_widgets/multi-topic/subsystem_widget.dart'; +import 'package:elastic_dashboard/widgets/nt4_widgets/multi-topic/swerve_drive.dart'; import 'package:elastic_dashboard/widgets/nt4_widgets/single_topic/match_time.dart'; import 'package:elastic_dashboard/widgets/nt4_widgets/single_topic/multi_color_view.dart'; import 'package:elastic_dashboard/widgets/nt4_widgets/single_topic/number_bar.dart'; @@ -356,6 +357,11 @@ class DraggableNT4WidgetContainer extends DraggableWidgetContainer { key: UniqueKey(), jsonData: jsonData['properties'], ); + case 'SwerveDrive': + return SwerveDriveWidget.fromJson( + key: UniqueKey(), + jsonData: jsonData['properties'], + ); case 'ComboBox Chooser': return ComboBoxChooser.fromJson( key: UniqueKey(), diff --git a/lib/widgets/network_tree/tree_row.dart b/lib/widgets/network_tree/tree_row.dart index 833ca369..3a4c0bec 100644 --- a/lib/widgets/network_tree/tree_row.dart +++ b/lib/widgets/network_tree/tree_row.dart @@ -14,6 +14,7 @@ import 'package:elastic_dashboard/widgets/nt4_widgets/multi-topic/power_distribu import 'package:elastic_dashboard/widgets/nt4_widgets/multi-topic/combo_box_chooser.dart'; import 'package:elastic_dashboard/widgets/nt4_widgets/multi-topic/robot_preferences.dart'; import 'package:elastic_dashboard/widgets/nt4_widgets/multi-topic/subsystem_widget.dart'; +import 'package:elastic_dashboard/widgets/nt4_widgets/multi-topic/swerve_drive.dart'; import 'package:elastic_dashboard/widgets/nt4_widgets/single_topic/boolean_box.dart'; import 'package:elastic_dashboard/widgets/nt4_widgets/nt4_widget.dart'; import 'package:elastic_dashboard/widgets/nt4_widgets/single_topic/text_display.dart'; @@ -159,6 +160,8 @@ class TreeRow { return PIDControllerWidget(key: UniqueKey(), topic: topic); case 'DifferentialDrive': return DifferentialDrive(key: UniqueKey(), topic: topic); + case 'SwerveDrive': + return SwerveDriveWidget(key: UniqueKey(), topic: topic); case 'String Chooser': return ComboBoxChooser(key: UniqueKey(), topic: topic); case 'Subsystem': @@ -205,6 +208,9 @@ class TreeRow { } else if (primary is DifferentialDrive) { width = normalGridSize * 3; height = normalGridSize * 2; + } else if (primary is SwerveDriveWidget) { + width = normalGridSize * 2; + height = normalGridSize * 2; } else if (primary is SubsystemWidget) { width = normalGridSize * 2; } else if (primary is CommandWidget) { diff --git a/lib/widgets/nt4_widgets/multi-topic/swerve_drive.dart b/lib/widgets/nt4_widgets/multi-topic/swerve_drive.dart new file mode 100644 index 00000000..af0be35a --- /dev/null +++ b/lib/widgets/nt4_widgets/multi-topic/swerve_drive.dart @@ -0,0 +1,564 @@ +import 'dart:math'; + +import 'package:dot_cast/dot_cast.dart'; +import 'package:elastic_dashboard/services/globals.dart'; +import 'package:elastic_dashboard/services/nt4_connection.dart'; +import 'package:elastic_dashboard/widgets/dialog_widgets/dialog_toggle_switch.dart'; +import 'package:elastic_dashboard/widgets/nt4_widgets/nt4_widget.dart'; +import 'package:flutter/material.dart'; +import 'package:provider/provider.dart'; +import 'package:vector_math/vector_math_64.dart' show radians, degrees; + +class SwerveDriveWidget extends StatelessWidget with NT4Widget { + @override + String type = 'SwerveDrive'; + + late String frontLeftAngleTopic; + late String frontLeftVelocityTopic; + + late String frontRightAngleTopic; + late String frontRightVelocityTopic; + + late String backLeftAngleTopic; + late String backLeftVelocityTopic; + + late String backRightAngleTopic; + late String backRightVelocityTopic; + + late String robotAngleTopic; + + bool showRobotRotation = true; + + SwerveDriveWidget({ + super.key, + required topic, + this.showRobotRotation = true, + period = Globals.defaultPeriod, + }) { + super.topic = topic; + super.period = period; + + init(); + } + + SwerveDriveWidget.fromJson( + {super.key, required Map jsonData}) { + topic = tryCast(jsonData['topic']) ?? ''; + period = tryCast(jsonData['period']) ?? Globals.defaultPeriod; + showRobotRotation = tryCast(jsonData['show_robot_rotation']) ?? true; + + init(); + } + + @override + void init() { + super.init(); + + frontLeftAngleTopic = '$topic/Front Left Angle'; + frontLeftVelocityTopic = '$topic/Front Left Velocity'; + + frontRightAngleTopic = '$topic/Front Right Angle'; + frontRightVelocityTopic = '$topic/Front Right Velocity'; + + backLeftAngleTopic = '$topic/Back Left Angle'; + backLeftVelocityTopic = '$topic/Back Left Velocity'; + + backRightAngleTopic = '$topic/Back Right Angle'; + backRightVelocityTopic = '$topic/Back Right Velocity'; + + robotAngleTopic = '$topic/Robot Angle'; + } + + @override + void resetSubscription() { + super.resetSubscription(); + + frontLeftAngleTopic = '$topic/Front Left Angle'; + frontLeftVelocityTopic = '$topic/Front Left Velocity'; + + frontRightAngleTopic = '$topic/Front Right Angle'; + frontRightVelocityTopic = '$topic/Front Right Velocity'; + + backLeftAngleTopic = '$topic/Back Left Angle'; + backLeftVelocityTopic = '$topic/Back Left Velocity'; + + backRightAngleTopic = '$topic/Back Right Angle'; + backRightVelocityTopic = '$topic/Back Right Velocity'; + + robotAngleTopic = '$topic/Robot Angle'; + } + + @override + Map toJson() { + return { + 'topic': topic, + 'period': period, + 'show_robot_rotation': showRobotRotation, + }; + } + + @override + List getEditProperties(BuildContext context) { + return [ + Center( + child: DialogToggleSwitch( + initialValue: showRobotRotation, + label: 'Show Robot Rotation', + onToggle: (value) { + showRobotRotation = value; + + refresh(); + }, + ), + ), + ]; + } + + @override + Widget build(BuildContext context) { + notifier = context.watch(); + + return StreamBuilder( + stream: subscription?.periodicStream(), + builder: (context, snapshot) { + double frontLeftAngle = + tryCast(nt4Connection.getLastAnnouncedValue(frontLeftAngleTopic)) ?? + 0.0; + double frontLeftVelocity = tryCast( + nt4Connection.getLastAnnouncedValue(frontLeftVelocityTopic)) ?? + 0.0; + + double frontRightAngle = tryCast( + nt4Connection.getLastAnnouncedValue(frontRightAngleTopic)) ?? + 0.0; + double frontRightVelocity = tryCast( + nt4Connection.getLastAnnouncedValue(frontRightVelocityTopic)) ?? + 0.0; + + double backLeftAngle = + tryCast(nt4Connection.getLastAnnouncedValue(backLeftAngleTopic)) ?? + 0.0; + double backLeftVelocity = tryCast( + nt4Connection.getLastAnnouncedValue(backLeftVelocityTopic)) ?? + 0.0; + + double backRightAngle = + tryCast(nt4Connection.getLastAnnouncedValue(backRightAngleTopic)) ?? + 0.0; + double backRightVelocity = tryCast( + nt4Connection.getLastAnnouncedValue(backRightVelocityTopic)) ?? + 0.0; + + double robotAngle = + tryCast(nt4Connection.getLastAnnouncedValue(robotAngleTopic)) ?? + 0.0; + + return LayoutBuilder(builder: (context, constraints) { + double sideLength = min(constraints.maxWidth, constraints.maxHeight); + return Transform.rotate( + angle: (showRobotRotation) ? radians(-robotAngle - 90) : 0.0, + child: SizedBox( + width: sideLength, + height: sideLength, + child: CustomPaint( + painter: SwerveDrivePainter( + frontLeftAngle: frontLeftAngle, + frontLeftVelocity: frontLeftVelocity, + frontRightAngle: frontRightAngle, + frontRightVelocity: frontRightVelocity, + backLeftAngle: backLeftAngle, + backLeftVelocity: backLeftVelocity, + backRightAngle: backRightAngle, + backRightVelocity: backRightVelocity, + ), + ), + ), + ); + }); + }, + ); + } +} + +class SwerveDrivePainter extends CustomPainter { + final double frontLeftAngle; + final double frontLeftVelocity; + + final double frontRightAngle; + final double frontRightVelocity; + + final double backLeftAngle; + final double backLeftVelocity; + + final double backRightAngle; + final double backRightVelocity; + + const SwerveDrivePainter({ + required this.frontLeftAngle, + required this.frontLeftVelocity, + required this.frontRightAngle, + required this.frontRightVelocity, + required this.backLeftAngle, + required this.backLeftVelocity, + required this.backRightAngle, + required this.backRightVelocity, + }); + + @override + void paint(Canvas canvas, Size size) { + const double robotFrameScale = 0.75; + const double arrowScale = robotFrameScale * 0.45; + + drawRobotFrame( + canvas, + size * robotFrameScale, + Offset(size.width - size.width * robotFrameScale, + size.height - size.height * robotFrameScale) / + 2); + + drawRobotDirectionArrow( + canvas, + size * arrowScale, + Offset(size.width - size.width * arrowScale, + size.height - size.height * arrowScale) / + 2); + + drawMotionArrows( + canvas, + size * robotFrameScale, + Offset(size.width - size.width * robotFrameScale, + size.height - size.height * robotFrameScale) / + 2); + } + + void drawRobotFrame(Canvas canvas, Size size, Offset offset) { + final double circleRadius = min(size.width, size.height) / 8; + + Paint framePainter = Paint() + ..strokeWidth = 2 + ..color = Colors.grey + ..style = PaintingStyle.stroke; + + // Front left circle + canvas.drawCircle(Offset(circleRadius, circleRadius) + offset, circleRadius, + framePainter); + + // Front right circle + canvas.drawCircle(Offset(size.width - circleRadius, circleRadius) + offset, + circleRadius, framePainter); + + // Back left circle + canvas.drawCircle(Offset(circleRadius, size.height - circleRadius) + offset, + circleRadius, framePainter); + + // Back right circle + canvas.drawCircle( + Offset(offset.dx + size.width - circleRadius, + offset.dy + size.height - circleRadius), + circleRadius, + framePainter); + + // Top line + canvas.drawLine( + Offset(circleRadius * 2, circleRadius) + offset, + Offset(size.width - circleRadius * 2, circleRadius) + offset, + framePainter); + + // Right line + canvas.drawLine( + Offset(size.width - circleRadius, circleRadius * 2) + offset, + Offset(size.width - circleRadius, size.height - circleRadius * 2) + + offset, + framePainter); + + // Bottom line + canvas.drawLine( + Offset(circleRadius * 2, size.height - circleRadius) + offset, + Offset(size.width - circleRadius * 2, size.height - circleRadius) + + offset, + framePainter); + + // Left line + canvas.drawLine( + Offset(circleRadius, circleRadius * 2) + offset, + Offset(circleRadius, size.height - circleRadius * 2) + offset, + framePainter); + } + + void drawMotionArrows(Canvas canvas, Size size, Offset offset) { + final double circleRadius = min(size.width, size.height) / 8; + const double arrowAngle = 40 * pi / 180; + + final double pixelsPerMPS = 12.5 / 1.0 * size.width / 240.0; + + const double minArrowBase = 8.5; + const double maxArrowBase = 16.0; + + Paint arrowPaint = Paint() + ..strokeWidth = 2 + ..color = Colors.red + ..style = PaintingStyle.stroke; + + Paint anglePaint = Paint() + ..strokeWidth = 4 + ..color = Colors.red + ..style = PaintingStyle.stroke; + + // Front left angle indicator thing + Rect frontLeftWheel = Rect.fromCenter( + center: Offset(circleRadius, circleRadius) + offset, + width: circleRadius * 2, + height: circleRadius * 2); + + canvas.drawArc(frontLeftWheel, radians(-(frontLeftAngle - 90 - 22.5)), + radians(-45), false, anglePaint); + + // Front left vector arrow + if (frontLeftVelocity.abs() >= 0.25) { + double frontLeftAngle = this.frontLeftAngle; + + frontLeftAngle -= 90; + frontLeftAngle *= -1; + + if (frontLeftVelocity < 0) { + frontLeftAngle -= 180; + } + + frontLeftAngle = radians(frontLeftAngle); + + double frontLeftArrowLength = frontLeftVelocity.abs() * pixelsPerMPS; + double frontLeftArrowBase = + (frontLeftArrowLength / 3.0).clamp(minArrowBase, maxArrowBase); + + canvas.drawLine( + Offset(circleRadius, circleRadius) + offset, + Offset(frontLeftArrowLength * cos(frontLeftAngle), + frontLeftArrowLength * sin(frontLeftAngle)) + + Offset(circleRadius, circleRadius) + + offset, + arrowPaint); + + drawArrowHead( + canvas, + Offset(circleRadius, circleRadius) / 2 + offset, + frontLeftArrowLength * cos(frontLeftAngle) + circleRadius / 2, + frontLeftArrowLength * sin(frontLeftAngle) + circleRadius / 2, + frontLeftAngle, + arrowAngle, + frontLeftArrowBase, + arrowPaint); + } else { + // Draw an X + drawX(canvas, Offset(circleRadius, circleRadius) + offset, circleRadius, + arrowPaint); + } + + // Front right angle indicator thing + Rect frontRightWheel = Rect.fromCenter( + center: Offset(size.width - circleRadius, circleRadius) + offset, + width: circleRadius * 2, + height: circleRadius * 2); + + canvas.drawArc(frontRightWheel, radians(-(frontRightAngle - 90 - 22.5)), + radians(-45), false, anglePaint); + + // Front right vector arrow + if (frontRightVelocity.abs() >= 0.25) { + double frontRightAngle = this.frontRightAngle; + + frontRightAngle -= 90; + frontRightAngle *= -1; + + if (frontRightVelocity < 0) { + frontRightAngle -= 180; + } + + frontRightAngle = radians(frontRightAngle); + + double frontRightArrowLength = frontRightVelocity.abs() * pixelsPerMPS; + double frontRightArrowBase = + (frontRightArrowLength / 3.0).clamp(minArrowBase, maxArrowBase); + + canvas.drawLine( + Offset(size.width - circleRadius, circleRadius) + offset, + Offset(frontRightArrowLength * cos(frontRightAngle), + frontRightArrowLength * sin(frontRightAngle)) + + Offset(size.width - circleRadius, circleRadius) + + offset, + arrowPaint); + + drawArrowHead( + canvas, + Offset(size.width - circleRadius / 2, circleRadius / 2) + offset, + frontRightArrowLength * cos(frontRightAngle) - circleRadius / 2, + frontRightArrowLength * sin(frontRightAngle) + circleRadius / 2, + frontRightAngle, + arrowAngle, + frontRightArrowBase, + arrowPaint); + } else { + // Draw an X + drawX(canvas, Offset(size.width - circleRadius, circleRadius) + offset, + circleRadius, arrowPaint); + } + + // Back left angle indicator thing + Rect backLeftWheel = Rect.fromCenter( + center: Offset(circleRadius, size.height - circleRadius) + offset, + width: circleRadius * 2, + height: circleRadius * 2); + + canvas.drawArc(backLeftWheel, radians(-(backLeftAngle - 90 - 22.5)), + radians(-45), false, anglePaint); + + // Back left vector arrow + if (backLeftVelocity.abs() >= 0.25) { + double backLeftAngle = this.backLeftAngle; + + backLeftAngle -= 90; + backLeftAngle *= -1; + + if (backLeftVelocity < 0) { + backLeftAngle -= 180; + } + + backLeftAngle = radians(backLeftAngle); + + double backLeftArrowLength = backLeftVelocity.abs() * pixelsPerMPS; + double backLeftArrowBase = + (backLeftArrowLength / 3.0).clamp(minArrowBase, maxArrowBase); + + canvas.drawLine( + Offset(circleRadius, size.height - circleRadius) + offset, + Offset(backLeftArrowLength * cos(backLeftAngle), + backLeftArrowLength * sin(backLeftAngle)) + + Offset(circleRadius, size.height - circleRadius) + + offset, + arrowPaint); + + drawArrowHead( + canvas, + Offset(circleRadius / 2, size.height - circleRadius / 2) + offset, + backLeftArrowLength * cos(backLeftAngle) + circleRadius / 2, + backLeftArrowLength * sin(backLeftAngle) - circleRadius / 2, + backLeftAngle, + arrowAngle, + backLeftArrowBase, + arrowPaint); + } else { + // Draw an X + drawX(canvas, Offset(circleRadius, size.height - circleRadius) + offset, + circleRadius, arrowPaint); + } + + // Back right angle indicator thing + Rect backRightWheel = Rect.fromCenter( + center: Offset(size.width - circleRadius, size.height - circleRadius) + + offset, + width: circleRadius * 2, + height: circleRadius * 2); + + canvas.drawArc(backRightWheel, radians(-(backRightAngle - 90 - 22.5)), + radians(-45), false, anglePaint); + + // Back right vector arrow + if (backRightVelocity.abs() >= 0.25) { + double backRightAngle = this.backRightAngle; + + backRightAngle -= 90; + backRightAngle *= -1; + + if (backRightVelocity < 0) { + backRightAngle -= 180; + } + + backRightAngle = radians(backRightAngle); + + double backRightArrowLength = backLeftVelocity.abs() * pixelsPerMPS; + double backRightArrowBase = + (backRightArrowLength / 3.0).clamp(minArrowBase, maxArrowBase); + + canvas.drawLine( + Offset(size.width - circleRadius, size.height - circleRadius) + + offset, + Offset(backRightArrowLength * cos(backRightAngle), + backRightArrowLength * sin(backRightAngle)) + + Offset(size.width - circleRadius, size.height - circleRadius) + + offset, + arrowPaint); + + drawArrowHead( + canvas, + Offset(size.width - circleRadius / 2, + size.height - circleRadius / 2) + + offset, + backRightArrowLength * cos(backRightAngle) - circleRadius / 2, + backRightArrowLength * sin(backRightAngle) - circleRadius / 2, + backRightAngle, + arrowAngle, + backRightArrowBase, + arrowPaint); + } else { + // Draw an X + drawX( + canvas, + Offset(size.width - circleRadius, size.height - circleRadius) + + offset, + circleRadius, + arrowPaint); + } + } + + void drawX(Canvas canvas, Offset offset, double circleRadius, Paint xPaint) { + canvas.drawLine(Offset(circleRadius / 2, circleRadius / 2) * 0.75 + offset, + -Offset(circleRadius / 2, circleRadius / 2) * 0.75 + offset, xPaint); + + canvas.drawLine( + -Offset(-circleRadius / 2, circleRadius / 2) * 0.75 + offset, + Offset(-circleRadius / 2, circleRadius / 2) * 0.75 + offset, + xPaint); + } + + void drawArrowHead(Canvas canvas, Offset center, double tipX, double tipY, + double arrowRotation, double arrowAngle, double base, Paint arrowPaint) { + Path arrowPath = Path() + ..moveTo(center.dx + tipX - base * cos(arrowRotation - arrowAngle), + center.dy + tipY - base * sin(arrowRotation - arrowAngle)) + ..lineTo(center.dx + tipX, center.dy + tipY) + ..lineTo(center.dx + tipX - base * cos(arrowRotation + arrowAngle), + center.dy + tipY - base * sin(arrowRotation + arrowAngle)); + + canvas.drawPath(arrowPath, arrowPaint); + } + + void drawRobotDirectionArrow(Canvas canvas, Size size, Offset offset) { + const double arrowAngle = 40 * pi / 180; + final double base = size.width * 0.45; + const double arrowRotation = -pi / 2; + const double tipX = 0; + final double tipY = -size.height / 2; + + Offset center = Offset(size.width, size.height) / 2 + offset; + + Paint arrowPainter = Paint() + ..strokeWidth = 4 + ..color = Colors.red + ..style = PaintingStyle.stroke; + + Path arrowHead = Path() + ..moveTo(center.dx + tipX - base * cos(arrowRotation - arrowAngle), + center.dy + tipY - base * sin(arrowRotation - arrowAngle)) + ..lineTo(center.dx + tipX, center.dy + tipY) + ..lineTo(center.dx + tipX - base * cos(arrowRotation + arrowAngle), + center.dy + tipY - base * sin(arrowRotation + arrowAngle)); + + canvas.drawPath(arrowHead, arrowPainter); + canvas.drawLine(Offset(tipX, tipY) + center, Offset(tipX, -tipY) + center, + arrowPainter); + } + + @override + bool shouldRepaint(covariant CustomPainter oldDelegate) { + return false; + } +}