From b63e8735d68ea9d6bfbc1e9d78c6fb924e40c6b2 Mon Sep 17 00:00:00 2001 From: DanPeled <98838880+DanPeled@users.noreply.github.com> Date: Tue, 2 Jul 2024 20:40:08 +0300 Subject: [PATCH 1/9] added swerve angle offset due to different types of configurations in libraries --- .../multi-topic/yagsl_swerve_drive.dart | 184 ++++++++++-------- 1 file changed, 107 insertions(+), 77 deletions(-) diff --git a/lib/widgets/nt_widgets/multi-topic/yagsl_swerve_drive.dart b/lib/widgets/nt_widgets/multi-topic/yagsl_swerve_drive.dart index 4c1cb01e..f5b883fb 100644 --- a/lib/widgets/nt_widgets/multi-topic/yagsl_swerve_drive.dart +++ b/lib/widgets/nt_widgets/multi-topic/yagsl_swerve_drive.dart @@ -1,5 +1,7 @@ import 'dart:math'; +import 'package:elastic_dashboard/services/text_formatter_builder.dart'; +import 'package:elastic_dashboard/widgets/dialog_widgets/dialog_text_input.dart'; import 'package:flutter/material.dart'; import 'package:dot_cast/dot_cast.dart'; @@ -14,6 +16,8 @@ class YAGSLSwerveDriveModel extends NTWidgetModel { @override String type = YAGSLSwerveDrive.widgetType; + double _angleOffset = 0.0; + String get measuredStatesTopic => '$topic/measuredStates'; String get desiredStatesTopic => '$topic/desiredStates'; String get robotRotationTopic => '$topic/robotRotation'; @@ -53,6 +57,7 @@ class YAGSLSwerveDriveModel extends NTWidgetModel { : super.fromJson(jsonData: jsonData) { _showRobotRotation = tryCast(jsonData['show_robot_rotation']) ?? true; _showDesiredStates = tryCast(jsonData['show_desired_states']) ?? true; + _angleOffset = tryCast(jsonData['angle_offset']) ?? 0.0; } @override @@ -61,11 +66,15 @@ class YAGSLSwerveDriveModel extends NTWidgetModel { ...super.toJson(), 'show_robot_rotation': _showRobotRotation, 'show_desired_states': _showDesiredStates, + 'angle_offset': _angleOffset, }; } @override List getEditProperties(BuildContext context) { + String rotationUnit = + tryCast(ntConnection.getLastAnnouncedValue(rotationUnitTopic)) ?? + 'radians'; return [ Row( children: [ @@ -89,6 +98,22 @@ class YAGSLSwerveDriveModel extends NTWidgetModel { ), ], ), + const SizedBox(height: 5), + Row( + children: [ + Flexible( + child: DialogTextInput( + onSubmit: (value) { + double? newOffset = double.tryParse(value); + _angleOffset = newOffset ?? 0.0; + }, + formatter: TextFormatterBuilder.decimalTextFormatter(), + label: 'Angle Offset ($rotationUnit)', + initialText: _angleOffset.toString(), + ), + ), + ], + ), ]; } @@ -115,6 +140,8 @@ class YAGSLSwerveDriveModel extends NTWidgetModel { double robotAngle = tryCast(ntConnection.getLastAnnouncedValue(robotRotationTopic)) ?? 0.0; + robotAngle += _angleOffset; + double maxSpeed = tryCast(ntConnection.getLastAnnouncedValue(maxSpeedTopic)) ?? 4.5; @@ -139,85 +166,88 @@ class YAGSLSwerveDrive extends NTWidget { Widget build(BuildContext context) { YAGSLSwerveDriveModel model = cast(context.watch()); - return StreamBuilder( - stream: model.multiTopicPeriodicStream, - builder: (context, snapshot) { - List measuredStatesRaw = tryCast(ntConnection - .getLastAnnouncedValue(model.measuredStatesTopic)) ?? - []; - List desiredStatesRaw = tryCast( - ntConnection.getLastAnnouncedValue(model.desiredStatesTopic)) ?? - []; - - List measuredStates = - measuredStatesRaw.whereType().toList(); - List desiredStates = - desiredStatesRaw.whereType().toList(); - - double width = tryCast( - ntConnection.getLastAnnouncedValue(model.robotWidthTopic)) ?? - 1.0; - double length = tryCast( - ntConnection.getLastAnnouncedValue(model.robotLengthTopic)) ?? - width; - - if (width <= 0.0) { - width = 1.0; - } - if (length <= 0.0) { - length = 0.0; - } - - double sizeRatio = min(length, width) / max(length, width); - double lengthWidthRatio = length / width; - - String rotationUnit = tryCast( - ntConnection.getLastAnnouncedValue(model.rotationUnitTopic)) ?? - 'radians'; - - double robotAngle = tryCast( - ntConnection.getLastAnnouncedValue(model.robotRotationTopic)) ?? - 0.0; - - if (rotationUnit == 'degrees') { - robotAngle = radians(robotAngle); - } else if (rotationUnit == 'rotations') { - robotAngle *= 2 * pi; - } - - double maxSpeed = - tryCast(ntConnection.getLastAnnouncedValue(model.maxSpeedTopic)) ?? - 4.5; - - if (maxSpeed <= 0.0) { - maxSpeed = 4.5; - } - - return LayoutBuilder( - builder: (context, constraints) { - double maxSideLength = - min(constraints.maxWidth, constraints.maxHeight) * - 0.9 * - sizeRatio; - return Transform.rotate( - angle: (model.showRobotRotation) ? -robotAngle : 0.0, - child: SizedBox( - width: maxSideLength / lengthWidthRatio, - height: maxSideLength * lengthWidthRatio, - child: CustomPaint( - painter: SwerveDrivePainter( - rotationUnit: rotationUnit, - maxSpeed: maxSpeed, - moduleStates: measuredStates, - desiredStates: - (model.showDesiredStates) ? desiredStates : [], + return GestureDetector( + behavior: HitTestBehavior.translucent, + child: StreamBuilder( + stream: model.multiTopicPeriodicStream, + builder: (context, snapshot) { + List measuredStatesRaw = tryCast(ntConnection + .getLastAnnouncedValue(model.measuredStatesTopic)) ?? + []; + List desiredStatesRaw = tryCast(ntConnection + .getLastAnnouncedValue(model.desiredStatesTopic)) ?? + []; + + List measuredStates = + measuredStatesRaw.whereType().toList(); + List desiredStates = + desiredStatesRaw.whereType().toList(); + + double width = tryCast( + ntConnection.getLastAnnouncedValue(model.robotWidthTopic)) ?? + 1.0; + double length = tryCast( + ntConnection.getLastAnnouncedValue(model.robotLengthTopic)) ?? + width; + + if (width <= 0.0) { + width = 1.0; + } + if (length <= 0.0) { + length = 0.0; + } + + double sizeRatio = min(length, width) / max(length, width); + double lengthWidthRatio = length / width; + + String rotationUnit = tryCast(ntConnection + .getLastAnnouncedValue(model.rotationUnitTopic)) ?? + 'radians'; + + double robotAngle = tryCast(ntConnection + .getLastAnnouncedValue(model.robotRotationTopic)) ?? + 0.0; + + if (rotationUnit == 'degrees') { + robotAngle = radians(robotAngle + model._angleOffset); + } else if (rotationUnit == 'rotations') { + robotAngle *= 2 * pi + model._angleOffset; + } + + double maxSpeed = tryCast( + ntConnection.getLastAnnouncedValue(model.maxSpeedTopic)) ?? + 4.5; + + if (maxSpeed <= 0.0) { + maxSpeed = 4.5; + } + + return LayoutBuilder( + builder: (context, constraints) { + double maxSideLength = + min(constraints.maxWidth, constraints.maxHeight) * + 0.9 * + sizeRatio; + return Transform.rotate( + angle: (model.showRobotRotation) ? -robotAngle : 0.0, + child: SizedBox( + width: maxSideLength / lengthWidthRatio, + height: maxSideLength * lengthWidthRatio, + child: CustomPaint( + painter: SwerveDrivePainter( + rotationUnit: rotationUnit, + maxSpeed: maxSpeed, + moduleStates: measuredStates, + desiredStates: + (model.showDesiredStates) ? desiredStates : [], + ), ), ), - ), - ); - }, - ); - }, + ); + }, + ); + }, + ), ); } } From a392585568e7ac8a4ba69d5a9deaf18f5cdfbe65 Mon Sep 17 00:00:00 2001 From: DanPeled <98838880+DanPeled@users.noreply.github.com> Date: Sat, 20 Jul 2024 12:36:58 +0300 Subject: [PATCH 2/9] rewrote some of the code to make easier to read --- .../multi-topic/yagsl_swerve_drive.dart | 104 ++++++++---------- 1 file changed, 43 insertions(+), 61 deletions(-) diff --git a/lib/widgets/nt_widgets/multi-topic/yagsl_swerve_drive.dart b/lib/widgets/nt_widgets/multi-topic/yagsl_swerve_drive.dart index f5b883fb..4e52780d 100644 --- a/lib/widgets/nt_widgets/multi-topic/yagsl_swerve_drive.dart +++ b/lib/widgets/nt_widgets/multi-topic/yagsl_swerve_drive.dart @@ -1,13 +1,10 @@ import 'dart:math'; - import 'package:elastic_dashboard/services/text_formatter_builder.dart'; import 'package:elastic_dashboard/widgets/dialog_widgets/dialog_text_input.dart'; import 'package:flutter/material.dart'; - import 'package:dot_cast/dot_cast.dart'; import 'package:provider/provider.dart'; import 'package:vector_math/vector_math_64.dart' show radians; - import 'package:elastic_dashboard/services/nt_connection.dart'; import 'package:elastic_dashboard/widgets/dialog_widgets/dialog_toggle_switch.dart'; import 'package:elastic_dashboard/widgets/nt_widgets/nt_widget.dart'; @@ -17,6 +14,24 @@ class YAGSLSwerveDriveModel extends NTWidgetModel { String type = YAGSLSwerveDrive.widgetType; double _angleOffset = 0.0; + bool _showRobotRotation; + bool _showDesiredStates; + + YAGSLSwerveDriveModel({ + required super.topic, + bool showRobotRotation = true, + bool showDesiredStates = true, + super.dataType, + super.period, + }) : _showDesiredStates = showDesiredStates, + _showRobotRotation = showRobotRotation, + super(); + + YAGSLSwerveDriveModel.fromJson({required Map jsonData}) + : _showRobotRotation = tryCast(jsonData['show_robot_rotation']) ?? true, + _showDesiredStates = tryCast(jsonData['show_desired_states']) ?? true, + _angleOffset = tryCast(jsonData['angle_offset']) ?? 0.0, + super.fromJson(jsonData: jsonData); String get measuredStatesTopic => '$topic/measuredStates'; String get desiredStatesTopic => '$topic/desiredStates'; @@ -26,40 +41,18 @@ class YAGSLSwerveDriveModel extends NTWidgetModel { String get robotLengthTopic => '$topic/sizeFrontBack'; String get rotationUnitTopic => '$topic/rotationUnit'; - bool _showRobotRotation = true; - bool _showDesiredStates = true; - - get showRobotRotation => _showRobotRotation; - - set showRobotRotation(value) { + bool get showRobotRotation => _showRobotRotation; + set showRobotRotation(bool value) { _showRobotRotation = value; refresh(); } - get showDesiredStates => _showDesiredStates; - - set showDesiredStates(value) { + bool get showDesiredStates => _showDesiredStates; + set showDesiredStates(bool value) { _showDesiredStates = value; refresh(); } - YAGSLSwerveDriveModel({ - required super.topic, - bool showRobotRotation = true, - bool showDesiredStates = true, - super.dataType, - super.period, - }) : _showDesiredStates = showDesiredStates, - _showRobotRotation = showRobotRotation, - super(); - - YAGSLSwerveDriveModel.fromJson({required Map jsonData}) - : super.fromJson(jsonData: jsonData) { - _showRobotRotation = tryCast(jsonData['show_robot_rotation']) ?? true; - _showDesiredStates = tryCast(jsonData['show_desired_states']) ?? true; - _angleOffset = tryCast(jsonData['angle_offset']) ?? 0.0; - } - @override Map toJson() { return { @@ -82,18 +75,14 @@ class YAGSLSwerveDriveModel extends NTWidgetModel { child: DialogToggleSwitch( initialValue: _showRobotRotation, label: 'Show Robot Rotation', - onToggle: (value) { - showRobotRotation = value; - }, + onToggle: (value) => showRobotRotation = value, ), ), Flexible( child: DialogToggleSwitch( initialValue: _showDesiredStates, label: 'Show Desired States', - onToggle: (value) { - showDesiredStates = value; - }, + onToggle: (value) => showDesiredStates = value, ), ), ], @@ -105,7 +94,9 @@ class YAGSLSwerveDriveModel extends NTWidgetModel { child: DialogTextInput( onSubmit: (value) { double? newOffset = double.tryParse(value); - _angleOffset = newOffset ?? 0.0; + if (newOffset != null) { + _angleOffset = newOffset; + } }, formatter: TextFormatterBuilder.decimalTextFormatter(), label: 'Angle Offset ($rotationUnit)', @@ -119,29 +110,27 @@ class YAGSLSwerveDriveModel extends NTWidgetModel { @override List getCurrentData() { - List measuredStatesRaw = + List rawMeasuredStates = tryCast(ntConnection.getLastAnnouncedValue(measuredStatesTopic)) ?? []; - List desiredStatesRaw = + List rawDesiredStates = tryCast(ntConnection.getLastAnnouncedValue(desiredStatesTopic)) ?? []; + // Filter and cast to List List measuredStates = - measuredStatesRaw.whereType().toList(); - List desiredStates = desiredStatesRaw.whereType().toList(); + List.from(rawMeasuredStates.whereType()); + List desiredStates = + List.from(rawDesiredStates.whereType()); double width = tryCast(ntConnection.getLastAnnouncedValue(robotWidthTopic)) ?? 1.0; double length = tryCast(ntConnection.getLastAnnouncedValue(robotLengthTopic)) ?? width; - String rotationUnit = tryCast(ntConnection.getLastAnnouncedValue(rotationUnitTopic)) ?? 'radians'; - double robotAngle = tryCast(ntConnection.getLastAnnouncedValue(robotRotationTopic)) ?? 0.0; - robotAngle += _angleOffset; - double maxSpeed = tryCast(ntConnection.getLastAnnouncedValue(maxSpeedTopic)) ?? 4.5; @@ -171,17 +160,18 @@ class YAGSLSwerveDrive extends NTWidget { child: StreamBuilder( stream: model.multiTopicPeriodicStream, builder: (context, snapshot) { - List measuredStatesRaw = tryCast(ntConnection + List rawMeasuredStates = tryCast(ntConnection .getLastAnnouncedValue(model.measuredStatesTopic)) ?? []; - List desiredStatesRaw = tryCast(ntConnection + List rawDesiredStates = tryCast(ntConnection .getLastAnnouncedValue(model.desiredStatesTopic)) ?? []; + // Filter and cast to List List measuredStates = - measuredStatesRaw.whereType().toList(); + List.from(rawMeasuredStates.whereType()); List desiredStates = - desiredStatesRaw.whereType().toList(); + List.from(rawDesiredStates.whereType()); double width = tryCast( ntConnection.getLastAnnouncedValue(model.robotWidthTopic)) ?? @@ -190,12 +180,8 @@ class YAGSLSwerveDrive extends NTWidget { ntConnection.getLastAnnouncedValue(model.robotLengthTopic)) ?? width; - if (width <= 0.0) { - width = 1.0; - } - if (length <= 0.0) { - length = 0.0; - } + width = width > 0.0 ? width : 1.0; + length = length > 0.0 ? length : 0.0; double sizeRatio = min(length, width) / max(length, width); double lengthWidthRatio = length / width; @@ -203,7 +189,6 @@ class YAGSLSwerveDrive extends NTWidget { String rotationUnit = tryCast(ntConnection .getLastAnnouncedValue(model.rotationUnitTopic)) ?? 'radians'; - double robotAngle = tryCast(ntConnection .getLastAnnouncedValue(model.robotRotationTopic)) ?? 0.0; @@ -217,10 +202,7 @@ class YAGSLSwerveDrive extends NTWidget { double maxSpeed = tryCast( ntConnection.getLastAnnouncedValue(model.maxSpeedTopic)) ?? 4.5; - - if (maxSpeed <= 0.0) { - maxSpeed = 4.5; - } + maxSpeed = maxSpeed > 0.0 ? maxSpeed : 4.5; return LayoutBuilder( builder: (context, constraints) { @@ -229,7 +211,7 @@ class YAGSLSwerveDrive extends NTWidget { 0.9 * sizeRatio; return Transform.rotate( - angle: (model.showRobotRotation) ? -robotAngle : 0.0, + angle: model.showRobotRotation ? -robotAngle : 0.0, child: SizedBox( width: maxSideLength / lengthWidthRatio, height: maxSideLength * lengthWidthRatio, @@ -239,7 +221,7 @@ class YAGSLSwerveDrive extends NTWidget { maxSpeed: maxSpeed, moduleStates: measuredStates, desiredStates: - (model.showDesiredStates) ? desiredStates : [], + model.showDesiredStates ? desiredStates : [], ), ), ), From bdc1ed86a1fcc993abf67957e9b870b6da8a8db3 Mon Sep 17 00:00:00 2001 From: DanPeled <98838880+DanPeled@users.noreply.github.com> Date: Sat, 20 Jul 2024 19:01:08 +0300 Subject: [PATCH 3/9] merged with original repo --- .github/workflows/elastic-ci.yml | 3 + lib/main.dart | 63 +- lib/pages/dashboard_page.dart | 469 +++++++++------ lib/services/hotkey_manager.dart | 8 + lib/services/nt4_client.dart | 158 ++--- lib/services/nt_connection.dart | 95 ++- lib/services/nt_widget_builder.dart | 40 +- .../robot_notifications_listener.dart | 4 +- lib/services/settings.dart | 32 +- lib/services/shuffleboard_nt_listener.dart | 121 ++-- lib/util/tab_data.dart | 11 + .../draggable_layout_container.dart | 8 +- .../draggable_list_layout.dart | 8 +- .../draggable_nt_widget_container.dart | 8 +- .../draggable_widget_container.dart | 65 +- .../models/layout_container_model.dart | 2 + .../models/list_layout_model.dart | 44 +- .../models/nt_widget_container_model.dart | 27 +- .../models/widget_container_model.dart | 48 +- lib/widgets/editable_tab_bar.dart | 65 +- .../network_tree/networktables_tree.dart | 39 +- .../network_tree/networktables_tree_row.dart | 38 +- .../nt_widgets/multi-topic/accelerometer.dart | 18 +- .../multi-topic/basic_swerve_drive.dart | 32 +- .../nt_widgets/multi-topic/camera_stream.dart | 22 +- .../multi-topic/combo_box_chooser.dart | 39 +- .../multi-topic/command_scheduler.dart | 24 +- .../multi-topic/command_widget.dart | 21 +- .../multi-topic/differential_drive.dart | 38 +- .../multi-topic/encoder_widget.dart | 27 +- .../nt_widgets/multi-topic/field_widget.dart | 21 +- .../nt_widgets/multi-topic/fms_info.dart | 45 +- lib/widgets/nt_widgets/multi-topic/gyro.dart | 24 +- .../multi-topic/motor_controller.dart | 18 +- .../multi-topic/network_alerts.dart | 22 +- .../multi-topic/pid_controller.dart | 44 +- .../multi-topic/power_distribution.dart | 32 +- .../multi-topic/profiled_pid_controller.dart | 48 +- .../nt_widgets/multi-topic/relay_widget.dart | 27 +- .../multi-topic/robot_preferences.dart | 48 +- .../multi-topic/split_button_chooser.dart | 30 +- .../multi-topic/subsystem_widget.dart | 20 +- .../multi-topic/three_axis_accelerometer.dart | 27 +- .../nt_widgets/multi-topic/ultrasonic.dart | 18 +- .../multi-topic/yagsl_swerve_drive.dart | 90 ++- lib/widgets/nt_widgets/nt_widget.dart | 29 +- .../nt_widgets/single_topic/boolean_box.dart | 30 +- .../nt_widgets/single_topic/graph.dart | 13 +- .../nt_widgets/single_topic/match_time.dart | 28 +- .../single_topic/multi_color_view.dart | 3 +- .../nt_widgets/single_topic/number_bar.dart | 22 +- .../single_topic/number_slider.dart | 20 +- .../nt_widgets/single_topic/radial_gauge.dart | 12 +- .../single_topic/single_color_view.dart | 3 +- .../nt_widgets/single_topic/text_display.dart | 19 +- .../single_topic/toggle_button.dart | 9 +- .../single_topic/toggle_switch.dart | 9 +- .../nt_widgets/single_topic/voltage_view.dart | 22 +- lib/widgets/settings_dialog.dart | 131 ++-- lib/widgets/tab_grid.dart | 224 ++++--- pubspec.yaml | 2 + test/main_test.dart | 5 +- test/pages/dashboard_page_test.dart | 384 +++++++++++- test/services/hotkey_manager_test.dart | 72 +++ test/services/nt_test.dart | 52 +- .../robot_notifications_listener_test.dart | 114 ++++ .../shuffleboard_nt_listener_test.dart | 37 +- test/test_util.dart | 104 +++- test/widgets/editable_tab_bar_test.dart | 189 ++++-- .../multi-topic/accelerometer_test.dart | 94 +++ .../multi-topic/basic_swerve_drive_test.dart | 106 ++++ .../multi-topic/camera_stream_test.dart | 141 +++++ .../multi-topic/combo_box_chooser_test.dart | 162 +++++ .../command_scheduler_widget_test.dart | 131 ++++ .../multi-topic/command_widget_test.dart | 124 ++++ .../multi-topic/differential_drive_test.dart | 122 ++++ .../multi-topic/encoder_widget_test.dart | 119 ++++ .../nt_widgets/multi-topic/fms_info_test.dart | 309 ++++++++++ .../nt_widgets/multi-topic/gyro_test.dart | 101 ++++ .../multi-topic/motor_controller_test.dart | 106 ++++ .../multi-topic/network_alerts_test.dart | 114 ++++ .../multi-topic/pid_controller_test.dart | 168 ++++++ .../multi-topic/power_distribution_test.dart | 138 +++++ .../profiled_pid_controller_test.dart | 167 ++++++ .../multi-topic/relay_widget_test.dart | 115 ++++ .../multi-topic/robot_preferences_test.dart | 158 +++++ .../split_button_chooser_test.dart | 127 ++++ .../multi-topic/subsystem_widget_test.dart | 97 +++ .../three_axis_accelerometer_test.dart | 123 ++++ .../multi-topic/ultrasonic_test.dart | 92 +++ .../multi-topic/yagsl_swerve_drive_test.dart | 90 +++ .../single-topic/boolean_box_test.dart | 150 +++++ .../nt_widgets/single-topic/graph_test.dart | 121 ++++ .../single-topic/match_time_test.dart | 156 +++++ .../single-topic/multi_color_view_test.dart | 215 +++++++ .../single-topic/number_bar_test.dart | 212 +++++++ .../single-topic/number_slider_test.dart | 205 +++++++ .../single-topic/radial_gauge_test.dart | 218 +++++++ .../single-topic/single_color_view_test.dart | 112 ++++ .../single-topic/text_display_test.dart | 564 ++++++++++++++++++ .../single-topic/toggle_button_test.dart | 114 ++++ .../single-topic/toggle_switch_test.dart | 114 ++++ .../single-topic/voltage_view_test.dart | 212 +++++++ test/widgets/settings_dialog_test.dart | 107 +++- test/widgets/tab_grid_test.dart | 49 +- test_resources/test-layout.json | 4 +- 106 files changed, 7869 insertions(+), 1212 deletions(-) create mode 100644 lib/util/tab_data.dart create mode 100644 test/services/hotkey_manager_test.dart create mode 100644 test/services/robot_notifications_listener_test.dart create mode 100644 test/widgets/nt_widgets/multi-topic/accelerometer_test.dart create mode 100644 test/widgets/nt_widgets/multi-topic/basic_swerve_drive_test.dart create mode 100644 test/widgets/nt_widgets/multi-topic/camera_stream_test.dart create mode 100644 test/widgets/nt_widgets/multi-topic/combo_box_chooser_test.dart create mode 100644 test/widgets/nt_widgets/multi-topic/command_scheduler_widget_test.dart create mode 100644 test/widgets/nt_widgets/multi-topic/command_widget_test.dart create mode 100644 test/widgets/nt_widgets/multi-topic/differential_drive_test.dart create mode 100644 test/widgets/nt_widgets/multi-topic/encoder_widget_test.dart create mode 100644 test/widgets/nt_widgets/multi-topic/fms_info_test.dart create mode 100644 test/widgets/nt_widgets/multi-topic/gyro_test.dart create mode 100644 test/widgets/nt_widgets/multi-topic/motor_controller_test.dart create mode 100644 test/widgets/nt_widgets/multi-topic/network_alerts_test.dart create mode 100644 test/widgets/nt_widgets/multi-topic/pid_controller_test.dart create mode 100644 test/widgets/nt_widgets/multi-topic/power_distribution_test.dart create mode 100644 test/widgets/nt_widgets/multi-topic/profiled_pid_controller_test.dart create mode 100644 test/widgets/nt_widgets/multi-topic/relay_widget_test.dart create mode 100644 test/widgets/nt_widgets/multi-topic/robot_preferences_test.dart create mode 100644 test/widgets/nt_widgets/multi-topic/split_button_chooser_test.dart create mode 100644 test/widgets/nt_widgets/multi-topic/subsystem_widget_test.dart create mode 100644 test/widgets/nt_widgets/multi-topic/three_axis_accelerometer_test.dart create mode 100644 test/widgets/nt_widgets/multi-topic/ultrasonic_test.dart create mode 100644 test/widgets/nt_widgets/multi-topic/yagsl_swerve_drive_test.dart create mode 100644 test/widgets/nt_widgets/single-topic/boolean_box_test.dart create mode 100644 test/widgets/nt_widgets/single-topic/graph_test.dart create mode 100644 test/widgets/nt_widgets/single-topic/match_time_test.dart create mode 100644 test/widgets/nt_widgets/single-topic/multi_color_view_test.dart create mode 100644 test/widgets/nt_widgets/single-topic/number_bar_test.dart create mode 100644 test/widgets/nt_widgets/single-topic/number_slider_test.dart create mode 100644 test/widgets/nt_widgets/single-topic/radial_gauge_test.dart create mode 100644 test/widgets/nt_widgets/single-topic/single_color_view_test.dart create mode 100644 test/widgets/nt_widgets/single-topic/text_display_test.dart create mode 100644 test/widgets/nt_widgets/single-topic/toggle_button_test.dart create mode 100644 test/widgets/nt_widgets/single-topic/toggle_switch_test.dart create mode 100644 test/widgets/nt_widgets/single-topic/voltage_view_test.dart diff --git a/.github/workflows/elastic-ci.yml b/.github/workflows/elastic-ci.yml index 836a34fa..2cc8aa1f 100644 --- a/.github/workflows/elastic-ci.yml +++ b/.github/workflows/elastic-ci.yml @@ -40,6 +40,9 @@ jobs: - name: Verify formatting run: dart format --output=none --set-exit-if-changed lib/* test/* + - name: Verify import sorting + run: dart run import_sorter:main --exit-if-changed + - name: Analyze project source run: flutter analyze --no-fatal-infos --no-fatal-warnings test: diff --git a/lib/main.dart b/lib/main.dart index 8e4917c8..8d968a70 100644 --- a/lib/main.dart +++ b/lib/main.dart @@ -4,6 +4,7 @@ import 'dart:math'; import 'package:flutter/material.dart'; +import 'package:collection/collection.dart'; import 'package:dot_cast/dot_cast.dart'; import 'package:flex_seed_scheme/flex_seed_scheme.dart'; import 'package:package_info_plus/package_info_plus.dart'; @@ -14,7 +15,6 @@ import 'package:window_manager/window_manager.dart'; import 'package:elastic_dashboard/pages/dashboard_page.dart'; import 'package:elastic_dashboard/services/field_images.dart'; -import 'package:elastic_dashboard/services/ip_address_util.dart'; import 'package:elastic_dashboard/services/log.dart'; import 'package:elastic_dashboard/services/nt_connection.dart'; import 'package:elastic_dashboard/services/nt_widget_builder.dart'; @@ -55,33 +55,12 @@ void main() async { await windowManager.ensureInitialized(); - Settings.teamNumber = - preferences.getInt(PrefKeys.teamNumber) ?? Settings.teamNumber; - Settings.ipAddressMode = - IPAddressMode.fromIndex(preferences.getInt(PrefKeys.ipAddressMode)); - - Settings.layoutLocked = - preferences.getBool(PrefKeys.layoutLocked) ?? Settings.layoutLocked; - Settings.gridSize = - preferences.getInt(PrefKeys.gridSize) ?? Settings.gridSize; - Settings.showGrid = - preferences.getBool(PrefKeys.showGrid) ?? Settings.showGrid; - Settings.cornerRadius = - preferences.getDouble(PrefKeys.cornerRadius) ?? Settings.cornerRadius; - Settings.autoResizeToDS = - preferences.getBool(PrefKeys.autoResizeToDS) ?? Settings.autoResizeToDS; - Settings.defaultPeriod = - preferences.getDouble(PrefKeys.defaultPeriod) ?? Settings.defaultPeriod; - Settings.defaultGraphPeriod = - preferences.getDouble(PrefKeys.defaultGraphPeriod) ?? - Settings.defaultGraphPeriod; - NTWidgetBuilder.ensureInitialized(); - Settings.ipAddress = - preferences.getString(PrefKeys.ipAddress) ?? Settings.ipAddress; + String ipAddress = + preferences.getString(PrefKeys.ipAddress) ?? Defaults.ipAddress; - ntConnection.nt4Connect(Settings.ipAddress); + NTConnection ntConnection = NTConnection(ipAddress); await FieldImages.loadFields('assets/fields/'); @@ -105,7 +84,13 @@ void main() async { await windowManager.show(); await windowManager.focus(); - runApp(Elastic(version: packageInfo.version, preferences: preferences)); + runApp( + Elastic( + ntConnection: ntConnection, + preferences: preferences, + version: packageInfo.version, + ), + ); } Future _restoreWindowPosition(SharedPreferences preferences, @@ -188,10 +173,15 @@ Future _restorePreferencesFromBackup(String appFolderPath) async { } class Elastic extends StatefulWidget { + final NTConnection ntConnection; final SharedPreferences preferences; final String version; - const Elastic({super.key, required this.version, required this.preferences}); + const Elastic( + {super.key, + required this.ntConnection, + required this.preferences, + required this.version}); @override State createState() => _ElasticState(); @@ -200,6 +190,11 @@ class Elastic extends StatefulWidget { class _ElasticState extends State { late Color teamColor = Color( widget.preferences.getInt(PrefKeys.teamColor) ?? Colors.blueAccent.value); + late FlexSchemeVariant themeVariant = FlexSchemeVariant.values + .firstWhereOrNull((element) => + element.variantName == + widget.preferences.getString(PrefKeys.themeVariant)) ?? + FlexSchemeVariant.material3Legacy; @override Widget build(BuildContext context) { @@ -208,7 +203,7 @@ class _ElasticState extends State { colorScheme: SeedColorScheme.fromSeeds( primaryKey: teamColor, brightness: Brightness.dark, - variant: FlexSchemeVariant.material3Legacy, + variant: themeVariant, ), ); return MaterialApp( @@ -216,12 +211,24 @@ class _ElasticState extends State { title: 'Elastic', theme: theme, home: DashboardPage( + ntConnection: widget.ntConnection, preferences: widget.preferences, version: widget.version, onColorChanged: (color) => setState(() { teamColor = color; widget.preferences.setInt(PrefKeys.teamColor, color.value); }), + onThemeVariantChanged: (variant) async { + themeVariant = variant; + if (variant == Defaults.themeVariant) { + await widget.preferences + .setString(PrefKeys.themeVariant, Defaults.defaultVariantName); + } else { + await widget.preferences + .setString(PrefKeys.themeVariant, variant.variantName); + } + setState(() {}); + }, ), ); } diff --git a/lib/pages/dashboard_page.dart b/lib/pages/dashboard_page.dart index 9cc5ab49..422a6db2 100644 --- a/lib/pages/dashboard_page.dart +++ b/lib/pages/dashboard_page.dart @@ -11,6 +11,7 @@ import 'package:dot_cast/dot_cast.dart'; import 'package:elegant_notification/elegant_notification.dart'; import 'package:elegant_notification/resources/stacked_options.dart'; import 'package:file_selector/file_selector.dart'; +import 'package:flex_seed_scheme/flex_seed_scheme.dart'; import 'package:popover/popover.dart'; import 'package:screen_retriever/screen_retriever.dart'; import 'package:shared_preferences/shared_preferences.dart'; @@ -25,6 +26,7 @@ import 'package:elastic_dashboard/services/robot_notifications_listener.dart'; import 'package:elastic_dashboard/services/settings.dart'; import 'package:elastic_dashboard/services/shuffleboard_nt_listener.dart'; import 'package:elastic_dashboard/services/update_checker.dart'; +import 'package:elastic_dashboard/util/tab_data.dart'; import 'package:elastic_dashboard/widgets/custom_appbar.dart'; import 'package:elastic_dashboard/widgets/dialog_widgets/dialog_toggle_switch.dart'; import 'package:elastic_dashboard/widgets/dialog_widgets/layout_drag_tile.dart'; @@ -37,15 +39,19 @@ import 'package:elastic_dashboard/widgets/tab_grid.dart'; import '../widgets/draggable_containers/models/layout_container_model.dart'; class DashboardPage extends StatefulWidget { - final SharedPreferences preferences; final String version; + final NTConnection ntConnection; + final SharedPreferences preferences; final Function(Color color)? onColorChanged; + final Function(FlexSchemeVariant variant)? onThemeVariantChanged; const DashboardPage({ super.key, + required this.ntConnection, required this.preferences, required this.version, this.onColorChanged, + this.onThemeVariantChanged, }); @override @@ -53,17 +59,16 @@ class DashboardPage extends StatefulWidget { } class _DashboardPageState extends State with WindowListener { - late final SharedPreferences _preferences; + late final SharedPreferences preferences = widget.preferences; late final UpdateChecker _updateChecker; late final RobotNotificationsListener _robotNotificationListener; - final List _grids = []; - final List _tabData = []; final Function _mapEquals = const DeepCollectionEquality().equals; - int _gridSize = Settings.gridSize; + late int _gridSize = + preferences.getInt(PrefKeys.gridSize) ?? Defaults.gridSize; int _currentTabIndex = 0; @@ -72,7 +77,6 @@ class _DashboardPageState extends State with WindowListener { @override void initState() { super.initState(); - _preferences = widget.preferences; _updateChecker = UpdateChecker(currentVersion: widget.version); windowManager.addListener(this); @@ -84,22 +88,25 @@ class _DashboardPageState extends State with WindowListener { _setupShortcuts(); - ntConnection.dsClientConnect( + widget.ntConnection.dsClientConnect( onIPAnnounced: (ip) async { - if (Settings.ipAddressMode != IPAddressMode.driverStation) { + if (preferences.getInt(PrefKeys.ipAddressMode) != + IPAddressMode.driverStation.index) { return; } - if (_preferences.getString(PrefKeys.ipAddress) != ip) { - await _preferences.setString(PrefKeys.ipAddress, ip); + if (preferences.getString(PrefKeys.ipAddress) != ip) { + await preferences.setString(PrefKeys.ipAddress, ip); } else { return; } - ntConnection.changeIPAddress(ip); + widget.ntConnection.changeIPAddress(ip); }, onDriverStationDockChanged: (docked) { - if (Settings.autoResizeToDS && docked) { + if ((preferences.getBool(PrefKeys.autoResizeToDS) ?? + Defaults.autoResizeToDS) && + docked) { _onDriverStationDocked(); } else { _onDriverStationUndocked(); @@ -107,23 +114,25 @@ class _DashboardPageState extends State with WindowListener { }, ); - ntConnection.addConnectedListener(() { + widget.ntConnection.addConnectedListener(() { setState(() { - for (TabGrid grid in _grids) { + for (TabGridModel grid in _tabData.map((e) => e.tabGrid)) { grid.onNTConnect(); } }); }); - ntConnection.addDisconnectedListener(() { + widget.ntConnection.addDisconnectedListener(() { setState(() { - for (TabGrid grid in _grids) { + for (TabGridModel grid in _tabData.map((e) => e.tabGrid)) { grid.onNTDisconnect(); } }); }); ShuffleboardNTListener apiListener = ShuffleboardNTListener( + ntConnection: widget.ntConnection, + preferences: widget.preferences, onTabChanged: (tab) { int? parsedTabIndex = int.tryParse(tab); @@ -149,7 +158,8 @@ class _DashboardPageState extends State with WindowListener { }); }, onTabCreated: (tab) { - if (Settings.layoutLocked) { + if (preferences.getBool(PrefKeys.layoutLocked) ?? + Defaults.layoutLocked) { return; } @@ -159,16 +169,18 @@ class _DashboardPageState extends State with WindowListener { return; } - _tabData.add(TabData(name: tab)); - _grids.add( - TabGrid( - key: GlobalKey(), + _tabData.add(TabData( + name: tab, + tabGrid: TabGridModel( + ntConnection: widget.ntConnection, + preferences: widget.preferences, onAddWidgetPressed: _displayAddWidgetDialog, ), - ); + )); }, onWidgetAdded: (widgetData) { - if (Settings.layoutLocked) { + if (preferences.getBool(PrefKeys.layoutLocked) ?? + Defaults.layoutLocked) { return; } // Needs to be done in case if widget data gets erased by the listener @@ -182,11 +194,16 @@ class _DashboardPageState extends State with WindowListener { String tabName = widgetDataCopy['tab']; if (!tabNamesList.contains(tabName)) { - _tabData.add(TabData(name: tabName)); - _grids.add(TabGrid( - key: GlobalKey(), - onAddWidgetPressed: _displayAddWidgetDialog, - )); + _tabData.add( + TabData( + name: tabName, + tabGrid: TabGridModel( + ntConnection: widget.ntConnection, + preferences: widget.preferences, + onAddWidgetPressed: _displayAddWidgetDialog, + ), + ), + ); tabNamesList.add(tabName); } @@ -197,7 +214,7 @@ class _DashboardPageState extends State with WindowListener { return; } - _grids[tabIndex].addWidgetFromTabJson(widgetDataCopy); + _tabData[tabIndex].tabGrid.addWidgetFromTabJson(widgetDataCopy); setState(() {}); }, @@ -211,7 +228,7 @@ class _DashboardPageState extends State with WindowListener { Future(() => _checkForUpdates(notifyIfLatest: false, notifyIfError: false)); _robotNotificationListener = RobotNotificationsListener( - connection: ntConnection, + ntConnection: widget.ntConnection, onNotification: (title, description, icon) { setState(() { ColorScheme colorScheme = Theme.of(context).colorScheme; @@ -246,7 +263,7 @@ class _DashboardPageState extends State with WindowListener { @override void onWindowClose() async { Map savedJson = - jsonDecode(_preferences.getString(PrefKeys.layout) ?? '{}'); + jsonDecode(preferences.getString(PrefKeys.layout) ?? '{}'); Map currentJson = _toJson(); bool showConfirmation = !_mapEquals(savedJson, currentJson); @@ -275,11 +292,10 @@ class _DashboardPageState extends State with WindowListener { for (int i = 0; i < _tabData.length; i++) { TabData data = _tabData[i]; - TabGrid grid = _grids[i]; gridData.add({ 'name': data.name, - 'grid_layout': grid.toJson(), + 'grid_layout': data.tabGrid.toJson(), }); } @@ -297,7 +313,7 @@ class _DashboardPageState extends State with WindowListener { TextTheme textTheme = Theme.of(context).textTheme; bool successful = - await _preferences.setString(PrefKeys.layout, jsonEncode(jsonData)); + await preferences.setString(PrefKeys.layout, jsonEncode(jsonData)); await _saveWindowPosition(); if (successful) { @@ -353,7 +369,7 @@ class _DashboardPageState extends State with WindowListener { String positionString = jsonEncode(positionArray); - await _preferences.setString(PrefKeys.windowPosition, positionString); + await preferences.setString(PrefKeys.windowPosition, positionString); } void _checkForUpdates( @@ -487,7 +503,7 @@ class _DashboardPageState extends State with WindowListener { } void _importLayout() async { - if (Settings.layoutLocked) { + if (preferences.getBool(PrefKeys.layoutLocked) ?? Defaults.layoutLocked) { return; } const XTypeGroup jsonTypeGroup = XTypeGroup( @@ -531,13 +547,13 @@ class _DashboardPageState extends State with WindowListener { return; } - await _preferences.setString(PrefKeys.layout, jsonEncode(jsonData)); + await preferences.setString(PrefKeys.layout, jsonEncode(jsonData)); setState(() => _loadLayoutFromJsonData(jsonString)); } void _loadLayout() { - String? jsonString = _preferences.getString(PrefKeys.layout); + String? jsonString = preferences.getString(PrefKeys.layout); if (jsonString == null) { _createDefaultTabs(); @@ -567,12 +583,10 @@ class _DashboardPageState extends State with WindowListener { if (jsonData.containsKey('grid_size')) { _gridSize = tryCast(jsonData['grid_size']) ?? _gridSize; - Settings.gridSize = _gridSize; - _preferences.setInt(PrefKeys.gridSize, _gridSize); + preferences.setInt(PrefKeys.gridSize, _gridSize); } _tabData.clear(); - _grids.clear(); for (Map data in jsonData['tabs']) { if (tryCast(data['name']) == null) { @@ -586,42 +600,47 @@ class _DashboardPageState extends State with WindowListener { continue; } - _tabData.add(TabData(name: data['name'])); - - _grids.add( - TabGrid.fromJson( - key: GlobalKey(), - jsonData: data['grid_layout'], - onAddWidgetPressed: _displayAddWidgetDialog, - onJsonLoadingWarning: _showJsonLoadingWarning, + _tabData.add( + TabData( + name: data['name'], + tabGrid: TabGridModel.fromJson( + ntConnection: widget.ntConnection, + preferences: widget.preferences, + jsonData: data['grid_layout'], + onAddWidgetPressed: _displayAddWidgetDialog, + onJsonLoadingWarning: _showJsonLoadingWarning, + ), ), ); } _createDefaultTabs(); - if (_currentTabIndex >= _grids.length) { - _currentTabIndex = _grids.length - 1; + if (_currentTabIndex >= _tabData.length) { + _currentTabIndex = _tabData.length - 1; } } void _createDefaultTabs() { - if (_tabData.isEmpty || _grids.isEmpty) { + if (_tabData.isEmpty) { logger.info('Creating default Teleoperated and Autonomous tabs'); setState(() { _tabData.addAll([ - TabData(name: 'Teleoperated'), - TabData(name: 'Autonomous'), - ]); - - _grids.addAll([ - TabGrid( - key: GlobalKey(), - onAddWidgetPressed: _displayAddWidgetDialog, + TabData( + name: 'Teleoperated', + tabGrid: TabGridModel( + ntConnection: widget.ntConnection, + preferences: widget.preferences, + onAddWidgetPressed: _displayAddWidgetDialog, + ), ), - TabGrid( - key: GlobalKey(), - onAddWidgetPressed: _displayAddWidgetDialog, + TabData( + name: 'Autonomous', + tabGrid: TabGridModel( + ntConnection: widget.ntConnection, + preferences: widget.preferences, + onAddWidgetPressed: _displayAddWidgetDialog, + ), ), ]); }); @@ -727,6 +746,30 @@ class _DashboardPageState extends State with WindowListener { }, ); } + // Move to next tab (Ctrl + Tab) + hotKeyManager.register( + HotKey( + LogicalKeyboardKey.tab, + modifiers: [KeyModifier.control], + ), + callback: () { + if (ModalRoute.of(context)?.isCurrent ?? false) { + _moveToNextTab(); + } + }, + ); + // Move to prevoius tab (Ctrl + Shift + Tab) + hotKeyManager.register( + HotKey( + LogicalKeyboardKey.tab, + modifiers: [KeyModifier.control, KeyModifier.shift], + ), + callback: () { + if (ModalRoute.of(context)?.isCurrent ?? false) { + _moveToPreviousTab(); + } + }, + ); // Move Tab Left (Ctrl + <-) hotKeyManager.register( HotKey( @@ -758,17 +801,21 @@ class _DashboardPageState extends State with WindowListener { modifiers: [KeyModifier.control], ), callback: () { - if (Settings.layoutLocked) { + if (preferences.getBool(PrefKeys.layoutLocked) ?? + Defaults.layoutLocked) { return; } String newTabName = 'Tab ${_tabData.length + 1}'; int newTabIndex = _tabData.length; - _tabData.add(TabData(name: newTabName)); - _grids.add( - TabGrid( - key: GlobalKey(), - onAddWidgetPressed: _displayAddWidgetDialog, + _tabData.add( + TabData( + name: newTabName, + tabGrid: TabGridModel( + ntConnection: widget.ntConnection, + preferences: widget.preferences, + onAddWidgetPressed: _displayAddWidgetDialog, + ), ), ); @@ -782,7 +829,8 @@ class _DashboardPageState extends State with WindowListener { modifiers: [KeyModifier.control], ), callback: () { - if (Settings.layoutLocked) { + if (preferences.getBool(PrefKeys.layoutLocked) ?? + Defaults.layoutLocked) { return; } if (_tabData.length <= 1) { @@ -798,11 +846,10 @@ class _DashboardPageState extends State with WindowListener { _currentTabIndex--; } - _grids[oldTabIndex].onDestroy(); + _tabData[oldTabIndex].tabGrid.onDestroy(); setState(() { _tabData.removeAt(oldTabIndex); - _grids.removeAt(oldTabIndex); }); }); }, @@ -810,19 +857,17 @@ class _DashboardPageState extends State with WindowListener { } void _lockLayout() async { - for (TabGrid grid in _grids) { + for (TabGridModel grid in _tabData.map((e) => e.tabGrid)) { grid.lockLayout(); } - Settings.layoutLocked = true; - await _preferences.setBool(PrefKeys.layoutLocked, true); + await preferences.setBool(PrefKeys.layoutLocked, true); } void _unlockLayout() async { - for (TabGrid grid in _grids) { + for (TabGridModel grid in _tabData.map((e) => e.tabGrid)) { grid.unlockLayout(); } - Settings.layoutLocked = false; - await _preferences.setBool(PrefKeys.layoutLocked, false); + await preferences.setBool(PrefKeys.layoutLocked, false); } void _displayAddWidgetDialog() { @@ -891,6 +936,7 @@ class _DashboardPageState extends State with WindowListener { showDialog( context: context, builder: (context) => SettingsDialog( + ntConnection: widget.ntConnection, preferences: widget.preferences, onTeamNumberChanged: (String? data) async { if (data == null) { @@ -900,15 +946,14 @@ class _DashboardPageState extends State with WindowListener { int? newTeamNumber = int.tryParse(data); if (newTeamNumber == null || - (newTeamNumber == Settings.teamNumber && - Settings.teamNumber != 9999)) { + (newTeamNumber == preferences.getInt(PrefKeys.teamNumber))) { return; } - await _preferences.setInt(PrefKeys.teamNumber, newTeamNumber); - Settings.teamNumber = newTeamNumber; + await preferences.setInt(PrefKeys.teamNumber, newTeamNumber); - switch (Settings.ipAddressMode) { + switch (IPAddressMode.fromIndex( + preferences.getInt(PrefKeys.ipAddressMode))) { case IPAddressMode.roboRIOmDNS: _updateIPAddress( IPAddressUtil.teamNumberToRIOmDNS(newTeamNumber)); @@ -922,16 +967,15 @@ class _DashboardPageState extends State with WindowListener { } }, onIPAddressModeChanged: (mode) async { - if (mode == Settings.ipAddressMode) { + if (mode.index == preferences.getInt(PrefKeys.ipAddressMode)) { return; } - await _preferences.setInt(PrefKeys.ipAddressMode, mode.index); - - Settings.ipAddressMode = mode; + await preferences.setInt(PrefKeys.ipAddressMode, mode.index); switch (mode) { case IPAddressMode.driverStation: - String? lastAnnouncedIP = ntConnection.dsClient.lastAnnouncedIP; + String? lastAnnouncedIP = + widget.ntConnection.dsClient.lastAnnouncedIP; if (lastAnnouncedIP == null) { break; @@ -940,12 +984,14 @@ class _DashboardPageState extends State with WindowListener { _updateIPAddress(lastAnnouncedIP); break; case IPAddressMode.roboRIOmDNS: - _updateIPAddress( - IPAddressUtil.teamNumberToRIOmDNS(Settings.teamNumber)); + _updateIPAddress(IPAddressUtil.teamNumberToRIOmDNS( + preferences.getInt(PrefKeys.teamNumber) ?? + Defaults.teamNumber)); break; case IPAddressMode.teamNumber: - _updateIPAddress( - IPAddressUtil.teamNumberToIP(Settings.teamNumber)); + _updateIPAddress(IPAddressUtil.teamNumberToIP( + preferences.getInt(PrefKeys.teamNumber) ?? + Defaults.teamNumber)); break; case IPAddressMode.localhost: _updateIPAddress('localhost'); @@ -956,18 +1002,17 @@ class _DashboardPageState extends State with WindowListener { } }, onIPAddressChanged: (String? data) async { - if (data == null || data == Settings.ipAddress) { + if (data == null || + data == preferences.getString(PrefKeys.ipAddress)) { return; } _updateIPAddress(data); }, onGridToggle: (value) async { - setState(() { - Settings.showGrid = value; - }); + await preferences.setBool(PrefKeys.showGrid, value); - await _preferences.setBool(PrefKeys.showGrid, value); + setState(() {}); }, onGridSizeChanged: (gridSize) async { if (gridSize == null) { @@ -1026,13 +1071,12 @@ class _DashboardPageState extends State with WindowListener { } setState(() { - Settings.gridSize = newGridSize; _gridSize = newGridSize; }); - await _preferences.setInt(PrefKeys.gridSize, newGridSize); + await preferences.setInt(PrefKeys.gridSize, newGridSize); - for (TabGrid grid in _grids) { + for (TabGridModel grid in _tabData.map((e) => e.tabGrid)) { grid.resizeGrid(_gridSize, _gridSize); } }, @@ -1043,44 +1087,40 @@ class _DashboardPageState extends State with WindowListener { double? newRadius = double.tryParse(radius); - if (newRadius == null || newRadius == Settings.cornerRadius) { + if (newRadius == null || + newRadius == preferences.getDouble(PrefKeys.cornerRadius)) { return; } - setState(() { - Settings.cornerRadius = newRadius; + await preferences.setDouble(PrefKeys.cornerRadius, newRadius); - for (TabGrid grid in _grids) { + setState(() { + for (TabGridModel grid in _tabData.map((e) => e.tabGrid)) { grid.refreshAllContainers(); } }); - - await _preferences.setDouble(PrefKeys.cornerRadius, newRadius); }, onResizeToDSChanged: (value) async { setState(() { - Settings.autoResizeToDS = value; - - if (value && ntConnection.dsClient.driverStationDocked) { + if (value && widget.ntConnection.dsClient.driverStationDocked) { _onDriverStationDocked(); } else { _onDriverStationUndocked(); } }); - await _preferences.setBool(PrefKeys.autoResizeToDS, value); + await preferences.setBool(PrefKeys.autoResizeToDS, value); }, onRememberWindowPositionChanged: (value) async { - await _preferences.setBool(PrefKeys.rememberWindowPosition, value); + await preferences.setBool(PrefKeys.rememberWindowPosition, value); }, onLayoutLock: (value) { - setState(() { - if (value) { - _lockLayout(); - } else { - _unlockLayout(); - } - }); + if (value) { + _lockLayout(); + } else { + _unlockLayout(); + } + setState(() {}); }, onDefaultPeriodChanged: (value) async { if (value == null) { @@ -1088,13 +1128,14 @@ class _DashboardPageState extends State with WindowListener { } double? newPeriod = double.tryParse(value); - if (newPeriod == null || newPeriod == Settings.defaultPeriod) { + if (newPeriod == null || + newPeriod == preferences.getDouble(PrefKeys.defaultPeriod)) { return; } - await _preferences.setDouble(PrefKeys.defaultPeriod, newPeriod); + await preferences.setDouble(PrefKeys.defaultPeriod, newPeriod); - setState(() => Settings.defaultPeriod = newPeriod); + setState(() {}); }, onDefaultGraphPeriodChanged: (value) async { if (value == null) { @@ -1102,28 +1143,29 @@ class _DashboardPageState extends State with WindowListener { } double? newPeriod = double.tryParse(value); - if (newPeriod == null || newPeriod == Settings.defaultGraphPeriod) { + if (newPeriod == null || + newPeriod == preferences.getDouble(PrefKeys.defaultGraphPeriod)) { return; } - await _preferences.setDouble(PrefKeys.defaultGraphPeriod, newPeriod); + await preferences.setDouble(PrefKeys.defaultGraphPeriod, newPeriod); - setState(() => Settings.defaultGraphPeriod = newPeriod); + setState(() {}); }, onColorChanged: widget.onColorChanged, + onThemeVariantChanged: widget.onThemeVariantChanged, ), ); } void _updateIPAddress(String newIPAddress) async { - if (newIPAddress == Settings.ipAddress) { + if (newIPAddress == preferences.getString(PrefKeys.ipAddress)) { return; } - await _preferences.setString(PrefKeys.ipAddress, newIPAddress); - Settings.ipAddress = newIPAddress; + await preferences.setString(PrefKeys.ipAddress, newIPAddress); setState(() { - ntConnection.changeIPAddress(newIPAddress); + widget.ntConnection.changeIPAddress(newIPAddress); }); } @@ -1226,7 +1268,7 @@ class _DashboardPageState extends State with WindowListener { } void _moveTabLeft() { - if (Settings.layoutLocked) { + if (preferences.getBool(PrefKeys.layoutLocked) ?? Defaults.layoutLocked) { return; } if (_currentTabIndex <= 0) { @@ -1238,22 +1280,17 @@ class _DashboardPageState extends State with WindowListener { logger.info('Moving current tab at index $_currentTabIndex to the left'); setState(() { - // Swap the tab data + // Swap the tabs TabData tempData = _tabData[_currentTabIndex - 1]; _tabData[_currentTabIndex - 1] = _tabData[_currentTabIndex]; _tabData[_currentTabIndex] = tempData; - // Swap the tab grids - TabGrid tempGrid = _grids[_currentTabIndex - 1]; - _grids[_currentTabIndex - 1] = _grids[_currentTabIndex]; - _grids[_currentTabIndex] = tempGrid; - _currentTabIndex -= 1; }); } void _moveTabRight() { - if (Settings.layoutLocked) { + if (preferences.getBool(PrefKeys.layoutLocked) ?? Defaults.layoutLocked) { return; } if (_currentTabIndex >= _tabData.length - 1) { @@ -1265,20 +1302,39 @@ class _DashboardPageState extends State with WindowListener { logger.info('Moving current tab at index $_currentTabIndex to the right'); setState(() { - // Swap the tab data + // Swap the tabs TabData tempData = _tabData[_currentTabIndex + 1]; _tabData[_currentTabIndex + 1] = _tabData[_currentTabIndex]; _tabData[_currentTabIndex] = tempData; - // Swap the tab grids - TabGrid tempGrid = _grids[_currentTabIndex + 1]; - _grids[_currentTabIndex + 1] = _grids[_currentTabIndex]; - _grids[_currentTabIndex] = tempGrid; - _currentTabIndex += 1; }); } + void _moveToNextTab() { + int moveIndex = _currentTabIndex + 1; + + if (moveIndex >= _tabData.length) { + moveIndex = 0; + } + + setState(() { + _currentTabIndex = moveIndex; + }); + } + + void _moveToPreviousTab() { + int moveIndex = _currentTabIndex - 1; + + if (moveIndex < 0) { + moveIndex = _tabData.length - 1; + } + + setState(() { + _currentTabIndex = moveIndex; + }); + } + @override Widget build(BuildContext context) { TextStyle? menuTextStyle = Theme.of(context).textTheme.bodySmall; @@ -1313,8 +1369,10 @@ class _DashboardPageState extends State with WindowListener { // Open Layout MenuItemButton( style: menuButtonStyle, - onPressed: - (!Settings.layoutLocked) ? () => _importLayout() : null, + onPressed: !(preferences.getBool(PrefKeys.layoutLocked) ?? + Defaults.layoutLocked) + ? () => _importLayout() + : null, shortcut: const SingleActivator(LogicalKeyboardKey.keyO, control: true), child: const Row( @@ -1372,10 +1430,13 @@ class _DashboardPageState extends State with WindowListener { // Clear layout MenuItemButton( style: menuButtonStyle, - onPressed: (!Settings.layoutLocked) + onPressed: !(preferences.getBool(PrefKeys.layoutLocked) ?? + Defaults.layoutLocked) ? () { setState(() { - _grids[_currentTabIndex].clearWidgets(context); + _tabData[_currentTabIndex] + .tabGrid + .clearWidgets(context); }); } : null, @@ -1386,19 +1447,21 @@ class _DashboardPageState extends State with WindowListener { MenuItemButton( style: menuButtonStyle, onPressed: () { - setState(() { - if (Settings.layoutLocked) { - _unlockLayout(); - } else { - _lockLayout(); - } - }); + if (preferences.getBool(PrefKeys.layoutLocked) ?? + Defaults.layoutLocked) { + _unlockLayout(); + } else { + _lockLayout(); + } + + setState(() {}); }, - leadingIcon: (Settings.layoutLocked) + leadingIcon: (preferences.getBool(PrefKeys.layoutLocked) ?? + Defaults.layoutLocked) ? const Icon(Icons.lock_open) : const Icon(Icons.lock_outline), child: Text( - '${(Settings.layoutLocked) ? 'Unlock' : 'Lock'} Layout'), + '${(preferences.getBool(PrefKeys.layoutLocked) ?? Defaults.layoutLocked) ? 'Unlock' : 'Lock'} Layout'), ) ], child: const Text( @@ -1458,11 +1521,14 @@ class _DashboardPageState extends State with WindowListener { MenuItemButton( style: menuButtonStyle, leadingIcon: const Icon(Icons.add), - onPressed: - (!Settings.layoutLocked) ? () => _displayAddWidgetDialog() : null, + onPressed: !(preferences.getBool(PrefKeys.layoutLocked) ?? + Defaults.layoutLocked) + ? () => _displayAddWidgetDialog() + : null, child: const Text('Add Widget'), ), - if (Settings.layoutLocked) ...[ + if ((preferences.getBool(PrefKeys.layoutLocked) ?? + Defaults.layoutLocked)) ...[ const VerticalDivider(), // Unlock Layout Tooltip( @@ -1475,9 +1541,8 @@ class _DashboardPageState extends State with WindowListener { const WidgetStatePropertyAll(Size(36.0, double.infinity)), ), onPressed: () { - setState(() { - _unlockLayout(); - }); + _unlockLayout(); + setState(() {}); }, child: const Icon(Icons.lock_outline), ), @@ -1503,6 +1568,7 @@ class _DashboardPageState extends State with WindowListener { child: Stack( children: [ EditableTabBar( + preferences: preferences, currentIndex: _currentTabIndex, onTabMoveLeft: () { _moveTabLeft(); @@ -1515,13 +1581,18 @@ class _DashboardPageState extends State with WindowListener { _tabData[index] = newData; }); }, - onTabCreate: (tab) { + onTabCreate: () { + String tabName = 'Tab ${_tabData.length + 1}'; setState(() { - _tabData.add(tab); - _grids.add(TabGrid( - key: GlobalKey(), - onAddWidgetPressed: _displayAddWidgetDialog, - )); + _tabData.add( + TabData( + name: tabName, + tabGrid: TabGridModel( + ntConnection: widget.ntConnection, + preferences: widget.preferences, + onAddWidgetPressed: _displayAddWidgetDialog, + )), + ); }); }, onTabDestroy: (index) { @@ -1536,50 +1607,60 @@ class _DashboardPageState extends State with WindowListener { _currentTabIndex--; } - _grids[index].onDestroy(); + _tabData[index].tabGrid.onDestroy(); setState(() { _tabData.removeAt(index); - _grids.removeAt(index); }); }); }, onTabChanged: (index) { setState(() => _currentTabIndex = index); }, - onTabDuplicate: (index, tab) { + onTabDuplicate: (index) { setState(() { - _tabData.insert(index + 1, tab); - Map tabJson = _grids[index].toJson(); - _grids.insert( + Map tabJson = + _tabData[index].tabGrid.toJson(); + TabGridModel newGrid = TabGridModel.fromJson( + ntConnection: widget.ntConnection, + preferences: preferences, + jsonData: tabJson, + onAddWidgetPressed: _displayAddWidgetDialog, + onJsonLoadingWarning: _showJsonLoadingWarning, + ); + _tabData.insert( index + 1, - TabGrid.fromJson( - key: GlobalKey(), - jsonData: tabJson, - onAddWidgetPressed: _displayAddWidgetDialog, - onJsonLoadingWarning: _showJsonLoadingWarning, - )); + TabData( + name: '${_tabData[index].name} (Copy)', + tabGrid: newGrid)); }); }, tabData: _tabData, - tabViews: _grids, ), _AddWidgetDialog( - grid: () => _grids[_currentTabIndex], + ntConnection: widget.ntConnection, + preferences: widget.preferences, + grid: () => _tabData[_currentTabIndex].tabGrid, visible: _addWidgetDialogVisible, onNTDragUpdate: (globalPosition, widget) { - _grids[_currentTabIndex] + _tabData[_currentTabIndex] + .tabGrid .addDragInWidget(widget, globalPosition); }, onNTDragEnd: (widget) { - _grids[_currentTabIndex].placeDragInWidget(widget); + _tabData[_currentTabIndex] + .tabGrid + .placeDragInWidget(widget); }, onLayoutDragUpdate: (globalPosition, widget) { - _grids[_currentTabIndex] + _tabData[_currentTabIndex] + .tabGrid .addDragInWidget(widget, globalPosition); }, onLayoutDragEnd: (widget) { - _grids[_currentTabIndex].placeDragInWidget(widget); + _tabData[_currentTabIndex] + .tabGrid + .placeDragInWidget(widget); }, onClose: () { setState(() => _addWidgetDialogVisible = false); @@ -1598,12 +1679,12 @@ class _DashboardPageState extends State with WindowListener { children: [ Expanded( child: StreamBuilder( - stream: ntConnection.connectionStatus(), + stream: widget.ntConnection.connectionStatus(), builder: (context, snapshot) { bool connected = snapshot.data ?? false; String connectedText = (connected) - ? 'Network Tables: Connected (${_preferences.getString(PrefKeys.ipAddress)})' + ? 'Network Tables: Connected (${preferences.getString(PrefKeys.ipAddress)})' : 'Network Tables: Disconnected'; return Text( @@ -1617,13 +1698,13 @@ class _DashboardPageState extends State with WindowListener { ), Expanded( child: Text( - 'Team ${_preferences.getInt(PrefKeys.teamNumber)?.toString() ?? 'Unknown'}', + 'Team ${preferences.getInt(PrefKeys.teamNumber)?.toString() ?? 'Unknown'}', textAlign: TextAlign.center, ), ), Expanded( child: StreamBuilder( - stream: ntConnection.latencyStream(), + stream: widget.ntConnection.latencyStream(), builder: (context, snapshot) { double latency = snapshot.data ?? 0.0; @@ -1645,7 +1726,9 @@ class _DashboardPageState extends State with WindowListener { } class _AddWidgetDialog extends StatefulWidget { - final TabGrid Function() _grid; + final NTConnection ntConnection; + final SharedPreferences preferences; + final TabGridModel Function() _grid; final bool _visible; final Function(Offset globalPosition, WidgetContainerModel widget) @@ -1659,7 +1742,9 @@ class _AddWidgetDialog extends StatefulWidget { final Function()? _onClose; const _AddWidgetDialog({ - required TabGrid Function() grid, + required this.ntConnection, + required this.preferences, + required TabGridModel Function() grid, required bool visible, required dynamic Function(Offset, WidgetContainerModel) onNTDragUpdate, required dynamic Function(WidgetContainerModel) onNTDragEnd, @@ -1715,6 +1800,8 @@ class _AddWidgetDialogState extends State<_AddWidgetDialog> { child: TabBarView( children: [ NetworkTableTree( + ntConnection: widget.ntConnection, + preferences: widget.preferences, listLayoutBuilder: ( {required title, required children}) { return widget._grid().createListLayout( diff --git a/lib/services/hotkey_manager.dart b/lib/services/hotkey_manager.dart index 27338947..e745f6dc 100644 --- a/lib/services/hotkey_manager.dart +++ b/lib/services/hotkey_manager.dart @@ -1,3 +1,4 @@ +import 'package:flutter/material.dart'; import 'package:flutter/services.dart'; import 'package:collection/collection.dart'; @@ -45,6 +46,13 @@ class HotKeyManager { _initialized = true; } + @visibleForTesting + void tearDown() { + _initialized = false; + _hotKeyList.clear(); + _callbackMap.clear(); + } + int _getNumberModifiersPressed() { int count = 0; diff --git a/lib/services/nt4_client.dart b/lib/services/nt4_client.dart index 4240f2d0..2189a502 100644 --- a/lib/services/nt4_client.dart +++ b/lib/services/nt4_client.dart @@ -58,8 +58,6 @@ class NT4Subscription { final NT4SubscriptionOptions options; final int uid; - int useCount = 0; - Object? currentValue; int timestamp = 0; @@ -153,7 +151,7 @@ class NT4Subscription { other.options == options; @override - int get hashCode => Object.hashAllUnordered([topic, options]); + int get hashCode => Object.hashAll([topic, options]); } class NT4SubscriptionOptions { @@ -189,7 +187,7 @@ class NT4SubscriptionOptions { @override int get hashCode => - Object.hashAllUnordered([periodicRateSeconds, all, topicsOnly, prefix]); + Object.hashAll([periodicRateSeconds, all, topicsOnly, prefix]); } class NT4Topic { @@ -272,7 +270,13 @@ class NT4Client { bool _serverConnectionActive = false; bool _rttConnectionActive = false; + bool get mainWebsocketActive => + _mainWebsocket != null && _mainWebsocket!.closeCode == null; + bool get rttWebsocketActive => + _rttWebsocket != null && _rttWebsocket!.closeCode == null; + bool _useRTT = false; + bool _attemptingConnection = true; int _lastPongTime = 0; @@ -292,7 +296,9 @@ class NT4Client { void setServerBaseAddreess(String serverBaseAddress) { this.serverBaseAddress = serverBaseAddress; - _wsOnClose(); + _wsOnClose(false); + _attemptingConnection = true; + Future.delayed(const Duration(milliseconds: 100), _connect); } Stream latencyStream() async* { @@ -322,23 +328,16 @@ class NT4Client { _topicAnnounceListeners.remove(onAnnounce); } - NT4Subscription subscribe(String topic, - [double period = 0.1, bool all = false]) { + NT4Subscription subscribe({ + required String topic, + NT4SubscriptionOptions options = const NT4SubscriptionOptions(), + }) { NT4Subscription newSub = NT4Subscription( topic: topic, uid: getNewSubUID(), - options: NT4SubscriptionOptions(periodicRateSeconds: period, all: all), + options: options, ); - if (_subscribedTopics.contains(newSub)) { - NT4Subscription subscription = _subscribedTopics.lookup(newSub)!; - subscription.useCount++; - - return subscription; - } - - newSub.useCount++; - _subscriptions[newSub.uid] = newSub; _subscribedTopics.add(newSub); _wsSubscribe(newSub); @@ -352,77 +351,28 @@ class NT4Client { return newSub; } - NT4Subscription subscribeAll(String topic, [double period = 0.1]) { - return subscribe(topic, period, true); - } - - NT4Subscription subscribeAllSamples(String topic, [double period = 0.1]) { - NT4Subscription newSub = NT4Subscription( - topic: topic, - uid: getNewSubUID(), - options: const NT4SubscriptionOptions(all: true), - ); - - if (_subscribedTopics.contains(newSub)) { - NT4Subscription subscription = _subscribedTopics.lookup(newSub)!; - subscription.useCount++; - - return subscription; - } - - newSub.useCount++; - - _subscriptions[newSub.uid] = newSub; - _wsSubscribe(newSub); - return newSub; - } - - NT4Subscription subscribeTopicsOnly(String topic) { - NT4Subscription newSub = NT4Subscription( - topic: topic, - uid: getNewSubUID(), - options: const NT4SubscriptionOptions(topicsOnly: true), - ); - - if (_subscribedTopics.contains(newSub)) { - NT4Subscription subscription = _subscribedTopics.lookup(newSub)!; - subscription.useCount++; - - return subscription; - } - - newSub.useCount++; - - _subscriptions[newSub.uid] = newSub; - _wsSubscribe(newSub); - return newSub; - } - void unSubscribe(NT4Subscription sub) { - sub.useCount--; - - if (sub.useCount <= 0) { - _subscriptions.remove(sub.uid); - _subscribedTopics.remove(sub); - _wsUnsubscribe(sub); - - // If there are no other subscriptions that are in the same table/tree - if (!_subscribedTopics.any((element) => - element.topic.startsWith('${sub.topic}/') || - '${sub.topic}/'.startsWith(element.topic))) { - // If there are any topics associated with the table/tree, unpublish them - for (NT4Topic topic in _clientPublishedTopics.values.where((element) => - element.name.startsWith('${sub.topic}/') || - '${sub.topic}/'.startsWith(element.name))) { - Future(() => unpublishTopic(topic)); - } + _subscriptions.remove(sub.uid); + _subscribedTopics.remove(sub); + _wsUnsubscribe(sub); + + // If there are no other subscriptions that are in the same table/tree + if (!_subscribedTopics.any((element) => + element.topic.startsWith('${sub.topic}/') || + sub.topic.startsWith('${element.topic}/') || + sub.topic == element.topic)) { + // If there are any topics associated with the table/tree, unpublish them + for (NT4Topic topic in _clientPublishedTopics.values.where((element) => + element.name.startsWith('${sub.topic}/') || + sub.topic.startsWith('${element.name}/') || + sub.topic == element.name)) { + Future(() => unpublishTopic(topic)); } } } void clearAllSubscriptions() { for (NT4Subscription sub in _subscriptions.values) { - sub.useCount = 0; unSubscribe(sub); } _subscriptions.clear(); @@ -506,8 +456,10 @@ class NT4Client { serialize([timeTopic.pubUID, 0, timeTopic.getTypeId(), timeToSend]); if (_useRTT) { - _rttWebsocket?.sink.add(rawData); - } else { + if (rttWebsocketActive && mainWebsocketActive) { + _rttWebsocket?.sink.add(rawData); + } + } else if (mainWebsocketActive) { _mainWebsocket?.sink.add(rawData); } } @@ -546,6 +498,10 @@ class NT4Client { } void _wsSendJSON(String method, Map params) { + if (!mainWebsocketActive) { + return; + } + _mainWebsocket?.sink.add(jsonEncode([ { 'method': method, @@ -555,11 +511,15 @@ class NT4Client { } void _wsSendBinary(dynamic data) { + if (!mainWebsocketActive) { + return; + } + _mainWebsocket?.sink.add(data); } void _connect() async { - if (_serverConnectionActive) { + if (_serverConnectionActive || !_attemptingConnection) { return; } @@ -578,10 +538,13 @@ class NT4Client { } catch (e) { // Failed to connect... try again logger.info( - 'Failed to connect to network tables, attempting to reconnect in 1 second'); - Future.delayed(const Duration(seconds: 1), _connect); + 'Failed to connect to network tables, attempting to reconnect in 500 ms'); + if (_attemptingConnection) { + Future.delayed(const Duration(milliseconds: 500), _connect); + } return; } + _attemptingConnection = false; if (!mainServerAddr.contains(serverBaseAddress)) { logger.info('IP Address changed while connecting, aborting connection'); @@ -596,7 +559,6 @@ class NT4Client { _useRTT = true; _pingInterval = _pingIntervalMsV41; _timeoutInterval = _pingTimeoutMsV41; - await _rttConnect(); } else { _useRTT = false; _pingInterval = _pingIntervalMsV40; @@ -619,6 +581,10 @@ class NT4Client { cancelOnError: true, ); + if (_useRTT) { + await _rttConnect(); + } + NT4Topic timeTopic = NT4Topic( name: 'Time', type: NT4TypeStr.kInt, @@ -659,8 +625,8 @@ class NT4Client { await _rttWebsocket!.ready; } catch (e) { logger.info( - 'Failed to connect to RTT Network Tables protocol, attempting to reconnect in 1 second'); - Future.delayed(const Duration(seconds: 1), _rttConnect); + 'Failed to connect to RTT Network Tables protocol, attempting to reconnect in 500 ms'); + Future.delayed(const Duration(milliseconds: 500), _rttConnect); return; } @@ -678,7 +644,7 @@ class NT4Client { _rttConnectionActive = true; } - if (!_serverConnectionActive && _mainWebsocket != null) { + if (!_serverConnectionActive && mainWebsocketActive) { _onFirstMessageReceived(); } @@ -736,7 +702,9 @@ class NT4Client { logger.info('RTT Connection closed'); } - void _wsOnClose() async { + void _wsOnClose([bool autoReconnect = true]) async { + _attemptingConnection = false; + _pingTimer?.cancel(); _pongTimer?.cancel(); @@ -764,9 +732,11 @@ class NT4Client { announcedTopics.clear(); - logger.debug('[NT4] Connection closed. Attempting to reconnect in 1s'); - - Future.delayed(const Duration(seconds: 1), _connect); + logger.debug('[NT4] Connection closed. Attempting to reconnect in 500 ms'); + if (autoReconnect && !_attemptingConnection) { + _attemptingConnection = true; + Future.delayed(const Duration(milliseconds: 500), _connect); + } } void _wsOnMessage(data) { diff --git a/lib/services/nt_connection.dart b/lib/services/nt_connection.dart index 79089ac5..f08dd861 100644 --- a/lib/services/nt_connection.dart +++ b/lib/services/nt_connection.dart @@ -3,11 +3,12 @@ import 'package:flutter/foundation.dart'; import 'package:elastic_dashboard/services/ds_interop.dart'; import 'package:elastic_dashboard/services/nt4_client.dart'; -NTConnection get ntConnection => NTConnection.instance; +typedef SubscriptionIdentification = ({ + String topic, + NT4SubscriptionOptions options +}); class NTConnection { - static NTConnection instance = NTConnection._internal(); - late NT4Client _ntClient; late DSInteropClient _dsClient; @@ -18,15 +19,21 @@ class NTConnection { bool _dsConnected = false; bool get isNT4Connected => _ntConnected; - NT4Client get nt4Client => _ntClient; bool get isDSConnected => _dsConnected; DSInteropClient get dsClient => _dsClient; - NTConnection._internal(); + @visibleForTesting + List get subscriptions => subscriptionUseCount.keys.toList(); + + @visibleForTesting + String get serverBaseAddress => _ntClient.serverBaseAddress; - factory NTConnection() { - return instance; + Map subscriptionMap = {}; + Map subscriptionUseCount = {}; + + NTConnection(String ipAddress) { + nt4Connect(ipAddress); } void nt4Connect(String ipAddress) { @@ -48,7 +55,10 @@ class NTConnection { }); // Allows all published topics to be announced - _ntClient.subscribeTopicsOnly('/'); + _ntClient.subscribe( + topic: '/', + options: const NT4SubscriptionOptions(topicsOnly: true), + ); } void dsClientConnect( @@ -70,6 +80,14 @@ class NTConnection { onDisconnectedListeners.add(callback); } + void addTopicAnnounceListener(Function(NT4Topic topic) onAnnounce) { + _ntClient.addTopicAnnounceListener(onAnnounce); + } + + void removeTopicAnnounceListener(Function(NT4Topic topic) onUnannounce) { + _ntClient.removeTopicAnnounceListener(onUnannounce); + } + Future? subscribeAndRetrieveData(String topic, {period = 0.1, timeout = const Duration(seconds: 2, milliseconds: 500)}) async { @@ -116,8 +134,12 @@ class NTConnection { } } + Map announcedTopics() { + return _ntClient.announcedTopics; + } + Stream latencyStream() { - return nt4Client.latencyStream(); + return _ntClient.latencyStream(); } void changeIPAddress(String ipAddress) { @@ -129,15 +151,51 @@ class NTConnection { } NT4Subscription subscribe(String topic, [double period = 0.1]) { - return _ntClient.subscribe(topic, period); + NT4SubscriptionOptions subscriptionOptions = + NT4SubscriptionOptions(periodicRateSeconds: period); + + int hashCode = Object.hash(topic, subscriptionOptions); + + if (subscriptionMap.containsKey(hashCode)) { + NT4Subscription existingSubscription = subscriptionMap[hashCode]!; + subscriptionUseCount.update(existingSubscription, (value) => value + 1); + + return existingSubscription; + } + + NT4Subscription newSubscription = + _ntClient.subscribe(topic: topic, options: subscriptionOptions); + + subscriptionMap[hashCode] = newSubscription; + subscriptionUseCount[newSubscription] = 1; + + return newSubscription; } NT4Subscription subscribeAll(String topic, [double period = 0.1]) { - return _ntClient.subscribeAll(topic, period); + return _ntClient.subscribe( + topic: topic, + options: NT4SubscriptionOptions( + periodicRateSeconds: period, + all: true, + )); } void unSubscribe(NT4Subscription subscription) { - _ntClient.unSubscribe(subscription); + if (!subscriptionUseCount.containsKey(subscription)) { + _ntClient.unSubscribe(subscription); + return; + } + + int hashCode = Object.hash(subscription.topic, subscription.options); + + subscriptionUseCount.update(subscription, (value) => value - 1); + + if (subscriptionUseCount[subscription]! <= 0) { + subscriptionMap.remove(hashCode); + subscriptionUseCount.remove(subscription); + _ntClient.unSubscribe(subscription); + } } NT4Topic? getTopicFromSubscription(NT4Subscription subscription) { @@ -148,6 +206,14 @@ class NTConnection { return _ntClient.getTopicFromName(topic); } + void publishTopic(NT4Topic topic) { + _ntClient.publishTopic(topic); + } + + NT4Topic publishNewTopic(String name, String type) { + return _ntClient.publishNewTopic(name, type); + } + bool isTopicPublished(NT4Topic? topic) { return _ntClient.isTopicPublished(topic); } @@ -167,4 +233,9 @@ class NTConnection { void updateDataFromTopic(NT4Topic topic, dynamic data) { _ntClient.addSample(topic, data); } + + @visibleForTesting + void updateDataFromTopicName(String topic, dynamic data) { + _ntClient.addSampleFromName(topic, data); + } } diff --git a/lib/services/nt_widget_builder.dart b/lib/services/nt_widget_builder.dart index d1df30e8..03d24d98 100644 --- a/lib/services/nt_widget_builder.dart +++ b/lib/services/nt_widget_builder.dart @@ -1,8 +1,10 @@ import 'package:flutter/material.dart'; import 'package:dot_cast/dot_cast.dart'; +import 'package:shared_preferences/shared_preferences.dart'; import 'package:elastic_dashboard/services/log.dart'; +import 'package:elastic_dashboard/services/nt_connection.dart'; import 'package:elastic_dashboard/services/settings.dart'; import 'package:elastic_dashboard/widgets/draggable_containers/draggable_widget_container.dart'; import 'package:elastic_dashboard/widgets/nt_widgets/multi-topic/accelerometer.dart'; @@ -49,6 +51,8 @@ class NTWidgetBuilder { static final Map< String, NTWidgetModel Function({ + required NTConnection ntConnection, + required SharedPreferences preferences, required String topic, String dataType, double period, @@ -57,6 +61,8 @@ class NTWidgetBuilder { static final Map< String, NTWidgetModel Function({ + required NTConnection ntConnection, + required SharedPreferences preferences, required Map jsonData, })> _modelJsonBuildMap = {}; @@ -68,6 +74,8 @@ class NTWidgetBuilder { static const double _normalSize = 128.0; + NTWidgetBuilder._(); + static bool _initialized = false; static void ensureInitialized() { if (_initialized) { @@ -141,6 +149,7 @@ class NTWidgetBuilder { SwerveDriveWidget.widgetType: BasicSwerveModel.fromJson, CameraStreamWidget.widgetType: CameraStreamModel.fromJson, ComboBoxChooser.widgetType: ComboBoxChooserModel.fromJson, + 'String Chooser': ComboBoxChooserModel.fromJson, CommandSchedulerWidget.widgetType: CommandSchedulerModel.fromJson, CommandWidget.widgetType: CommandModel.fromJson, DifferentialDrive.widgetType: DifferentialDriveModel.fromJson, @@ -320,17 +329,22 @@ class NTWidgetBuilder { } static NTWidgetModel buildNTModelFromType( + NTConnection ntConnection, + SharedPreferences preferences, String type, String topic, { String dataType = 'Unknown', double? period, }) { - period ??= Settings.defaultPeriod; + period ??= + preferences.getDouble(PrefKeys.defaultPeriod) ?? Defaults.defaultPeriod; ensureInitialized(); if (_modelNameBuildMap.containsKey(type)) { return _modelNameBuildMap[type]!( + ntConnection: ntConnection, + preferences: preferences, topic: topic, dataType: dataType, period: period, @@ -338,6 +352,8 @@ class NTWidgetBuilder { } return NTWidgetModel.createDefault( + ntConnection: ntConnection, + preferences: preferences, type: type, topic: topic, dataType: dataType, @@ -346,17 +362,27 @@ class NTWidgetBuilder { } static NTWidgetModel buildNTModelFromJson( - String type, Map jsonData, - {Function(String message)? onWidgetTypeNotFound}) { + NTConnection ntConnection, + SharedPreferences preferences, + String type, + Map jsonData, { + Function(String message)? onWidgetTypeNotFound, + }) { ensureInitialized(); if (_modelJsonBuildMap.containsKey(type)) { - return _modelJsonBuildMap[type]!(jsonData: jsonData); + return _modelJsonBuildMap[type]!( + ntConnection: ntConnection, + preferences: preferences, + jsonData: jsonData, + ); } onWidgetTypeNotFound ?.call('Unknown widget type: \'$type\', defaulting to Empty Model.'); return NTWidgetModel.createDefault( + ntConnection: ntConnection, + preferences: preferences, type: type, topic: tryCast(jsonData['topic']) ?? '', dataType: tryCast(jsonData['data_type']) ?? 'Unknown', @@ -370,7 +396,8 @@ class NTWidgetBuilder { if (_minimumWidthMap.containsKey(widget.type)) { return _minimumWidthMap[widget.type]!; } else { - return Settings.gridSize.toDouble(); + return (widget.preferences.getInt(PrefKeys.gridSize) ?? Defaults.gridSize) + .toDouble(); } } @@ -380,7 +407,8 @@ class NTWidgetBuilder { if (_minimumHeightMap.containsKey(widget.type)) { return _minimumHeightMap[widget.type]!; } else { - return Settings.gridSize.toDouble(); + return (widget.preferences.getInt(PrefKeys.gridSize) ?? Defaults.gridSize) + .toDouble(); } } diff --git a/lib/services/robot_notifications_listener.dart b/lib/services/robot_notifications_listener.dart index d61b955b..4de7adb7 100644 --- a/lib/services/robot_notifications_listener.dart +++ b/lib/services/robot_notifications_listener.dart @@ -8,11 +8,11 @@ import 'package:elastic_dashboard/services/nt_connection.dart'; class RobotNotificationsListener { bool _alertFirstRun = true; - final NTConnection connection; + final NTConnection ntConnection; final Function(String title, String description, Icon icon) onNotification; RobotNotificationsListener({ - required this.connection, + required this.ntConnection, required this.onNotification, }); diff --git a/lib/services/settings.dart b/lib/services/settings.dart index f85ed4c2..60945529 100644 --- a/lib/services/settings.dart +++ b/lib/services/settings.dart @@ -1,3 +1,5 @@ +import 'package:flex_seed_scheme/flex_seed_scheme.dart'; + import 'package:elastic_dashboard/services/ip_address_util.dart'; class Settings { @@ -5,23 +7,28 @@ class Settings { 'https://github.com/Gold872/elastic-dashboard'; static const String releasesLink = '$repositoryLink/releases/latest'; - static IPAddressMode ipAddressMode = IPAddressMode.driverStation; - - static String ipAddress = '127.0.0.1'; - static int teamNumber = 9999; - static int gridSize = 128; - static bool layoutLocked = false; - static double cornerRadius = 15.0; - static bool showGrid = true; - static bool autoResizeToDS = false; - // window_manager doesn't support drag disable/maximize // disable on some platforms, this is a dumb workaround for it static bool isWindowDraggable = true; static bool isWindowMaximizable = true; +} + +class Defaults { + static IPAddressMode ipAddressMode = IPAddressMode.driverStation; + + static FlexSchemeVariant themeVariant = FlexSchemeVariant.material3Legacy; + static const String defaultVariantName = 'Material-3 Legacy (Default)'; + + static const String ipAddress = '127.0.0.1'; + static const int teamNumber = 9999; + static const int gridSize = 128; + static const bool layoutLocked = false; + static const double cornerRadius = 15.0; + static const bool showGrid = true; + static const bool autoResizeToDS = false; - static double defaultPeriod = 0.06; - static double defaultGraphPeriod = 0.033; + static const double defaultPeriod = 0.06; + static const double defaultGraphPeriod = 0.033; } class PrefKeys { @@ -30,6 +37,7 @@ class PrefKeys { static String ipAddressMode = 'ip_address_mode'; static String teamNumber = 'team_number'; static String teamColor = 'team_color'; + static String themeVariant = 'theme_variant'; static String layoutLocked = 'layout_locked'; static String gridSize = 'grid_size'; static String cornerRadius = 'corner_radius'; diff --git a/lib/services/shuffleboard_nt_listener.dart b/lib/services/shuffleboard_nt_listener.dart index cf1714fb..0a7eefa8 100644 --- a/lib/services/shuffleboard_nt_listener.dart +++ b/lib/services/shuffleboard_nt_listener.dart @@ -1,4 +1,5 @@ import 'package:dot_cast/dot_cast.dart'; +import 'package:shared_preferences/shared_preferences.dart'; import 'package:elastic_dashboard/services/nt4_client.dart'; import 'package:elastic_dashboard/services/nt_connection.dart'; @@ -13,6 +14,8 @@ class ShuffleboardNTListener { static const String tabsEntry = '$metadataTable/Tabs'; static const String selectedEntry = '$metadataTable/Selected'; + final NTConnection ntConnection; + final SharedPreferences preferences; final Function(Map widgetData)? onWidgetAdded; final Function(String tab)? onTabChanged; final Function(String tab)? onTabCreated; @@ -23,11 +26,19 @@ class ShuffleboardNTListener { Map> currentJsonData = {}; - final NetworkTableTreeRow shuffleboardTreeRoot = - NetworkTableTreeRow(topic: '/', rowName: ''); + late final NetworkTableTreeRow shuffleboardTreeRoot = NetworkTableTreeRow( + ntConnection: ntConnection, + preferences: preferences, + topic: '/', + rowName: ''); - ShuffleboardNTListener( - {this.onTabChanged, this.onTabCreated, this.onWidgetAdded}); + ShuffleboardNTListener({ + required this.ntConnection, + required this.preferences, + this.onTabChanged, + this.onTabCreated, + this.onWidgetAdded, + }); void initializeSubscriptions() { selectedSubscription = ntConnection.subscribe(selectedEntry); @@ -60,7 +71,7 @@ class ShuffleboardNTListener { previousSelection = null; }); - ntConnection.nt4Client.addTopicAnnounceListener((topic) async { + ntConnection.addTopicAnnounceListener((topic) async { if (!topic.name.contains(shuffleboardTableRoot)) { return; } @@ -108,10 +119,6 @@ class ShuffleboardNTListener { Object? subProperty = await ntConnection.subscribeAndRetrieveData(propertyTopic); - if (subProperty == null) { - return; - } - String real = realHierarchy[realHierarchy.length - 1]; List realTopics = real.split('/'); bool inLayout = real.split('/').length > 6; @@ -187,10 +194,6 @@ class ShuffleboardNTListener { String? type = await ntConnection.subscribeAndRetrieveData(componentTopic); - if (type == null) { - return; - } - if (inLayout) { Map child = _createOrGetChild(jsonKey, widgetName); @@ -215,13 +218,27 @@ class ShuffleboardNTListener { if (inLayout) { Map child = _createOrGetChild(jsonKey, widgetName); - child.putIfAbsent('width', () => size[0] * Settings.gridSize); - child.putIfAbsent('height', () => size[1] * Settings.gridSize); + child.putIfAbsent( + 'width', + () => + size[0] * + (preferences.getInt(PrefKeys.gridSize) ?? Defaults.gridSize)); + child.putIfAbsent( + 'height', + () => + size[1] * + (preferences.getInt(PrefKeys.gridSize) ?? Defaults.gridSize)); } else { - currentJsonData[jsonKey]! - .putIfAbsent('width', () => size[0] * Settings.gridSize); - currentJsonData[jsonKey]! - .putIfAbsent('height', () => size[1] * Settings.gridSize); + currentJsonData[jsonKey]!.putIfAbsent( + 'width', + () => + size[0] * + (preferences.getInt(PrefKeys.gridSize) ?? Defaults.gridSize)); + currentJsonData[jsonKey]!.putIfAbsent( + 'height', + () => + size[1] * + (preferences.getInt(PrefKeys.gridSize) ?? Defaults.gridSize)); } } @@ -240,13 +257,27 @@ class ShuffleboardNTListener { if (inLayout) { Map child = _createOrGetChild(jsonKey, widgetName); - child.putIfAbsent('x', () => position[0] * Settings.gridSize); - child.putIfAbsent('y', () => position[1] * Settings.gridSize); + child.putIfAbsent( + 'x', + () => + position[0] * + (preferences.getInt(PrefKeys.gridSize) ?? Defaults.gridSize)); + child.putIfAbsent( + 'y', + () => + position[1] * + (preferences.getInt(PrefKeys.gridSize) ?? Defaults.gridSize)); } else { - currentJsonData[jsonKey]! - .putIfAbsent('x', () => position[0] * Settings.gridSize); - currentJsonData[jsonKey]! - .putIfAbsent('y', () => position[1] * Settings.gridSize); + currentJsonData[jsonKey]!.putIfAbsent( + 'x', + () => + position[0] * + (preferences.getInt(PrefKeys.gridSize) ?? Defaults.gridSize)); + currentJsonData[jsonKey]!.putIfAbsent( + 'y', + () => + position[1] * + (preferences.getInt(PrefKeys.gridSize) ?? Defaults.gridSize)); } } } @@ -305,10 +336,6 @@ class ShuffleboardNTListener { String? type = await ntConnection.subscribeAndRetrieveData(typeTopic, timeout: const Duration(seconds: 3)); - if (type == null) { - return; - } - if (type == 'ShuffleboardLayout') { currentJsonData[jsonKey]!['layout'] = true; } @@ -375,12 +402,14 @@ class ShuffleboardNTListener { 'width', () => (!isCameraStream) ? widget!.displayRect.width - : Settings.gridSize * 2); + : (preferences.getInt(PrefKeys.gridSize) ?? Defaults.gridSize) * + 2); currentJsonData[jsonKey]!.putIfAbsent( 'height', () => (!isCameraStream) ? widget!.displayRect.height - : Settings.gridSize * 2); + : (preferences.getInt(PrefKeys.gridSize) ?? Defaults.gridSize) * + 2); currentJsonData[jsonKey]!.putIfAbsent('tab', () => tabName); currentJsonData[jsonKey]!.putIfAbsent('type', () => type); currentJsonData[jsonKey]! @@ -390,8 +419,10 @@ class ShuffleboardNTListener { currentJsonData[jsonKey]!['properties'].putIfAbsent( 'period', () => (type != 'Graph') - ? Settings.defaultPeriod - : Settings.defaultGraphPeriod); + ? preferences.getDouble(PrefKeys.defaultPeriod) ?? + Defaults.defaultPeriod + : preferences.getDouble(PrefKeys.defaultGraphPeriod) ?? + Defaults.defaultGraphPeriod); if (ntConnection.isNT4Connected) { onWidgetAdded?.call(currentJsonData[jsonKey]!); @@ -421,10 +452,14 @@ class ShuffleboardNTListener { currentJsonData[jsonKey]!.putIfAbsent('title', () => componentName); currentJsonData[jsonKey]!.putIfAbsent('x', () => 0.0); currentJsonData[jsonKey]!.putIfAbsent('y', () => 0.0); - currentJsonData[jsonKey]! - .putIfAbsent('width', () => Settings.gridSize.toDouble()); - currentJsonData[jsonKey]! - .putIfAbsent('height', () => Settings.gridSize.toDouble()); + currentJsonData[jsonKey]!.putIfAbsent( + 'width', + () => (preferences.getInt(PrefKeys.gridSize) ?? Defaults.gridSize) + .toDouble()); + currentJsonData[jsonKey]!.putIfAbsent( + 'height', + () => (preferences.getInt(PrefKeys.gridSize) ?? Defaults.gridSize) + .toDouble()); currentJsonData[jsonKey]!.putIfAbsent('type', () => 'List Layout'); currentJsonData[jsonKey]!.putIfAbsent('tab', () => tabName); currentJsonData[jsonKey]! @@ -483,19 +518,23 @@ class ShuffleboardNTListener { 'width', () => (!isCameraStream) ? widget!.displayRect.width - : Settings.gridSize * 2); + : (preferences.getInt(PrefKeys.gridSize) ?? Defaults.gridSize) * + 2); child.putIfAbsent( 'height', () => (!isCameraStream) ? widget!.displayRect.height - : Settings.gridSize * 2); + : (preferences.getInt(PrefKeys.gridSize) ?? Defaults.gridSize) * + 2); child['properties']!.putIfAbsent('topic', () => childRow.topic); child['properties']!.putIfAbsent( 'period', () => (type != 'Graph') - ? Settings.defaultPeriod - : Settings.defaultGraphPeriod); + ? preferences.getDouble(PrefKeys.defaultPeriod) ?? + Defaults.defaultPeriod + : preferences.getDouble(PrefKeys.defaultGraphPeriod) ?? + Defaults.defaultGraphPeriod); widget?.unSubscribe(); widget?.disposeModel(deleting: true); diff --git a/lib/util/tab_data.dart b/lib/util/tab_data.dart new file mode 100644 index 00000000..78dc7cfd --- /dev/null +++ b/lib/util/tab_data.dart @@ -0,0 +1,11 @@ +import 'package:elastic_dashboard/widgets/tab_grid.dart'; + +class TabData { + String name; + TabGridModel tabGrid; + + TabData({ + required this.name, + required this.tabGrid, + }); +} diff --git a/lib/widgets/draggable_containers/draggable_layout_container.dart b/lib/widgets/draggable_containers/draggable_layout_container.dart index 0bfa1064..13e13008 100644 --- a/lib/widgets/draggable_containers/draggable_layout_container.dart +++ b/lib/widgets/draggable_containers/draggable_layout_container.dart @@ -3,12 +3,6 @@ import 'package:elastic_dashboard/widgets/draggable_containers/draggable_widget_ abstract class DraggableLayoutContainer extends DraggableWidgetContainer { const DraggableLayoutContainer({ super.key, - required super.tabGrid, - super.onUpdate, - super.onDragBegin, - super.onDragEnd, - super.onDragCancel, - super.onResizeBegin, - super.onResizeEnd, + super.updateFunctions, }) : super(); } diff --git a/lib/widgets/draggable_containers/draggable_list_layout.dart b/lib/widgets/draggable_containers/draggable_list_layout.dart index 41e6db90..a1725666 100644 --- a/lib/widgets/draggable_containers/draggable_list_layout.dart +++ b/lib/widgets/draggable_containers/draggable_list_layout.dart @@ -8,13 +8,7 @@ import 'models/list_layout_model.dart'; class DraggableListLayout extends DraggableLayoutContainer { const DraggableListLayout({ super.key, - required super.tabGrid, - super.onUpdate, - super.onDragBegin, - super.onDragEnd, - super.onDragCancel, - super.onResizeBegin, - super.onResizeEnd, + super.updateFunctions, }) : super(); @override diff --git a/lib/widgets/draggable_containers/draggable_nt_widget_container.dart b/lib/widgets/draggable_containers/draggable_nt_widget_container.dart index ec850baa..00b6c910 100644 --- a/lib/widgets/draggable_containers/draggable_nt_widget_container.dart +++ b/lib/widgets/draggable_containers/draggable_nt_widget_container.dart @@ -8,13 +8,7 @@ import 'models/nt_widget_container_model.dart'; class DraggableNTWidgetContainer extends DraggableWidgetContainer { const DraggableNTWidgetContainer({ super.key, - required super.tabGrid, - super.onUpdate, - super.onDragBegin, - super.onDragEnd, - super.onDragCancel, - super.onResizeBegin, - super.onResizeEnd, + super.updateFunctions, }) : super(); @override diff --git a/lib/widgets/draggable_containers/draggable_widget_container.dart b/lib/widgets/draggable_containers/draggable_widget_container.dart index 51bd874c..cbfb5446 100644 --- a/lib/widgets/draggable_containers/draggable_widget_container.dart +++ b/lib/widgets/draggable_containers/draggable_widget_container.dart @@ -4,35 +4,30 @@ import 'package:flutter_box_transform/flutter_box_transform.dart'; import 'package:provider/provider.dart'; import 'package:elastic_dashboard/services/settings.dart'; -import 'package:elastic_dashboard/widgets/tab_grid.dart'; import 'models/widget_container_model.dart'; +typedef DraggableContainerUpdateFunctions = ({ + Function(WidgetContainerModel widget, Rect newRect, + TransformResult result) onUpdate, + Function(WidgetContainerModel widget) onDragBegin, + Function(WidgetContainerModel widget, Rect releaseRect, + {Offset? globalPosition}) onDragEnd, + Function(WidgetContainerModel widget) onDragCancel, + Function(WidgetContainerModel widget) onResizeBegin, + Function(WidgetContainerModel widget, Rect releaseRect) onResizeEnd, + bool Function(WidgetContainerModel widget, Rect location) isValidMoveLocation, +}); + class DraggableWidgetContainer extends StatelessWidget { - final TabGrid tabGrid; - - final Function( - WidgetContainerModel widget, Rect newRect, TransformResult result)? - onUpdate; - final Function(WidgetContainerModel widget)? onDragBegin; - final Function(WidgetContainerModel widget, Rect releaseRect, - {Offset? globalPosition})? onDragEnd; - final Function(WidgetContainerModel widget)? onDragCancel; - final Function(WidgetContainerModel widget)? onResizeBegin; - final Function(WidgetContainerModel widget, Rect releaseRect)? onResizeEnd; + final DraggableContainerUpdateFunctions? updateFunctions; const DraggableWidgetContainer({ super.key, - required this.tabGrid, - this.onUpdate, - this.onDragBegin, - this.onDragEnd, - this.onDragCancel, - this.onResizeBegin, - this.onResizeEnd, + this.updateFunctions, }); static double snapToGrid(double value, [double? gridSize]) { - gridSize ??= Settings.gridSize.toDouble(); + gridSize ??= Defaults.gridSize.toDouble(); return (value / gridSize).roundToDouble() * gridSize; } @@ -65,9 +60,10 @@ class DraggableWidgetContainer extends StatelessWidget { model.setDragStartLocation(model.displayRect); model.setPreviewRect(model.dragStartLocation); model.setValidLocation( - tabGrid.isValidMoveLocation(model, model.previewRect)); + updateFunctions?.isValidMoveLocation(model, model.previewRect) ?? + true); - onDragBegin?.call(model); + updateFunctions?.onDragBegin(model); controller?.setRect(model.draggingRect); }, @@ -79,21 +75,22 @@ class DraggableWidgetContainer extends StatelessWidget { model.setDragStartLocation(model.displayRect); model.setPreviewRect(model.dragStartLocation); model.setValidLocation( - tabGrid.isValidMoveLocation(model, model.previewRect)); + updateFunctions?.isValidMoveLocation(model, model.previewRect) ?? + true); - onResizeBegin?.call(model); + updateFunctions?.onResizeBegin.call(model); controller?.setRect(model.draggingRect); }, onChanged: (result, event) { if (!model.dragging && !model.resizing) { - onDragCancel?.call(model); + updateFunctions?.onDragCancel(model); return; } model.setCursorGlobalLocation(event.globalPosition); - onUpdate?.call(model, result.rect, result); + updateFunctions?.onUpdate(model, result.rect, result); controller?.setRect(model.draggingRect); }, @@ -103,7 +100,7 @@ class DraggableWidgetContainer extends StatelessWidget { } model.setDragging(false); - onDragEnd?.call(model, model.draggingRect, + updateFunctions?.onDragEnd(model, model.draggingRect, globalPosition: model.cursorGlobalLocation); controller?.setRect(model.draggingRect); @@ -113,7 +110,7 @@ class DraggableWidgetContainer extends StatelessWidget { model.setDragging(false); }); - onDragCancel?.call(model); + updateFunctions?.onDragCancel(model); controller?.setRect(model.draggingRect); }, @@ -124,7 +121,7 @@ class DraggableWidgetContainer extends StatelessWidget { model.setDragging(false); model.setResizing(false); - onResizeEnd?.call(model, model.draggingRect); + updateFunctions?.onResizeEnd(model, model.draggingRect); controller?.setRect(model.draggingRect); }, @@ -132,7 +129,7 @@ class DraggableWidgetContainer extends StatelessWidget { model.setDragging(false); model.setResizing(false); - onDragCancel?.call(model); + updateFunctions?.onDragCancel(model); controller?.setRect(model.draggingRect); }, @@ -160,6 +157,7 @@ class WidgetContainer extends StatelessWidget { this.opacity = 1.0, this.horizontalPadding = 7.5, this.verticalPadding = 7.5, + this.cornerRadius = Defaults.cornerRadius, }); final double opacity; @@ -169,6 +167,7 @@ class WidgetContainer extends StatelessWidget { final double height; final double horizontalPadding; final double verticalPadding; + final double cornerRadius; @override Widget build(BuildContext context) { @@ -183,7 +182,7 @@ class WidgetContainer extends StatelessWidget { opacity: opacity, child: Container( decoration: BoxDecoration( - borderRadius: BorderRadius.circular(Settings.cornerRadius), + borderRadius: BorderRadius.circular(cornerRadius), color: const Color.fromARGB(255, 40, 40, 40), boxShadow: const [ BoxShadow( @@ -203,8 +202,8 @@ class WidgetContainer extends StatelessWidget { return Container( decoration: BoxDecoration( borderRadius: BorderRadius.only( - topLeft: Radius.circular(Settings.cornerRadius), - topRight: Radius.circular(Settings.cornerRadius), + topLeft: Radius.circular(cornerRadius), + topRight: Radius.circular(cornerRadius), ), color: theme.colorScheme.primaryContainer, ), diff --git a/lib/widgets/draggable_containers/models/layout_container_model.dart b/lib/widgets/draggable_containers/models/layout_container_model.dart index 8c73c654..7e61d707 100644 --- a/lib/widgets/draggable_containers/models/layout_container_model.dart +++ b/lib/widgets/draggable_containers/models/layout_container_model.dart @@ -9,6 +9,7 @@ abstract class LayoutContainerModel extends WidgetContainerModel { String get type; LayoutContainerModel({ + required super.preferences, required super.initialPosition, required super.title, super.minWidth, @@ -17,6 +18,7 @@ abstract class LayoutContainerModel extends WidgetContainerModel { LayoutContainerModel.fromJson({ required super.jsonData, + required super.preferences, super.enabled, super.minWidth, super.minHeight, diff --git a/lib/widgets/draggable_containers/models/list_layout_model.dart b/lib/widgets/draggable_containers/models/list_layout_model.dart index 4872a713..9ee32ae6 100644 --- a/lib/widgets/draggable_containers/models/list_layout_model.dart +++ b/lib/widgets/draggable_containers/models/list_layout_model.dart @@ -6,6 +6,7 @@ import 'package:flutter/material.dart'; import 'package:collection/collection.dart'; import 'package:dot_cast/dot_cast.dart'; import 'package:provider/provider.dart'; +import 'package:shared_preferences/shared_preferences.dart'; import 'package:elastic_dashboard/services/settings.dart'; import 'package:elastic_dashboard/widgets/dialog_widgets/dialog_dropdown_chooser.dart'; @@ -25,7 +26,14 @@ class ListLayoutModel extends LayoutContainerModel { String labelPosition = 'TOP'; - final TabGrid tabGrid; + final TabGridModel tabGrid; + final NTWidgetContainerModel? Function( + SharedPreferences preferences, + Map jsonData, + bool enabled, { + Function(String errorMessage)? onJsonLoadingWarning, + })? ntWidgetBuilder; + final Function(WidgetContainerModel model)? onDragCancel; static List labelPositions = const [ @@ -37,10 +45,12 @@ class ListLayoutModel extends LayoutContainerModel { ]; ListLayoutModel({ + required super.preferences, required super.initialPosition, required super.title, required this.tabGrid, required this.onDragCancel, + this.ntWidgetBuilder, List? children, super.minWidth, super.minHeight, @@ -53,6 +63,8 @@ class ListLayoutModel extends LayoutContainerModel { ListLayoutModel.fromJson({ required super.jsonData, + required super.preferences, + required this.ntWidgetBuilder, required this.tabGrid, required this.onDragCancel, super.enabled, @@ -119,13 +131,13 @@ class ListLayoutModel extends LayoutContainerModel { } for (Map childData in jsonData['children']) { - children.add( - NTWidgetContainerModel.fromJson( - jsonData: childData, - enabled: enabled, - onJsonLoadingWarning: onJsonLoadingWarning, - ), - ); + NTWidgetContainerModel? widgetModel = ntWidgetBuilder!( + preferences, childData, enabled, + onJsonLoadingWarning: onJsonLoadingWarning); + + if (widgetModel != null) { + children.add(widgetModel); + } } } @@ -454,7 +466,8 @@ class ListLayoutModel extends LayoutContainerModel { .whereNot((element) => element == PointerDeviceKind.trackpad) .toSet(), onPanDown: (details) { - if (Settings.layoutLocked) { + if (preferences.getBool(PrefKeys.layoutLocked) ?? + Defaults.layoutLocked) { return; } widget.cursorGlobalLocation = details.globalPosition; @@ -468,7 +481,8 @@ class ListLayoutModel extends LayoutContainerModel { }); }, onPanUpdate: (details) { - if (Settings.layoutLocked) { + if (preferences.getBool(PrefKeys.layoutLocked) ?? + Defaults.layoutLocked) { return; } widget.cursorGlobalLocation = details.globalPosition; @@ -479,7 +493,8 @@ class ListLayoutModel extends LayoutContainerModel { tabGrid.layoutDragOutUpdate(widget, location); }, onPanEnd: (details) { - if (Settings.layoutLocked) { + if (preferences.getBool(PrefKeys.layoutLocked) ?? + Defaults.layoutLocked) { return; } Future(() => setDraggable(true)); @@ -502,7 +517,8 @@ class ListLayoutModel extends LayoutContainerModel { tabGrid.layoutDragOutEnd(widget); }, onPanCancel: () { - if (Settings.layoutLocked) { + if (preferences.getBool(PrefKeys.layoutLocked) ?? + Defaults.layoutLocked) { return; } Future(() { @@ -545,6 +561,8 @@ class ListLayoutModel extends LayoutContainerModel { title: title, width: draggingRect.width, height: draggingRect.height, + cornerRadius: + preferences.getDouble(PrefKeys.cornerRadius) ?? Defaults.cornerRadius, opacity: 0.80, horizontalPadding: 5.0, verticalPadding: 5.0, @@ -572,6 +590,8 @@ class ListLayoutModel extends LayoutContainerModel { title: title, width: displayRect.width, height: displayRect.height, + cornerRadius: + preferences.getDouble(PrefKeys.cornerRadius) ?? Defaults.cornerRadius, opacity: (previewVisible) ? 0.25 : 1.00, horizontalPadding: 5.0, verticalPadding: 5.0, diff --git a/lib/widgets/draggable_containers/models/nt_widget_container_model.dart b/lib/widgets/draggable_containers/models/nt_widget_container_model.dart index 26138adc..0dd78022 100644 --- a/lib/widgets/draggable_containers/models/nt_widget_container_model.dart +++ b/lib/widgets/draggable_containers/models/nt_widget_container_model.dart @@ -4,6 +4,7 @@ import 'package:dot_cast/dot_cast.dart'; import 'package:flutter_context_menu/flutter_context_menu.dart'; import 'package:provider/provider.dart'; +import 'package:elastic_dashboard/services/nt_connection.dart'; import 'package:elastic_dashboard/services/nt_widget_builder.dart'; import 'package:elastic_dashboard/services/settings.dart'; import 'package:elastic_dashboard/services/text_formatter_builder.dart'; @@ -15,10 +16,13 @@ import 'package:elastic_dashboard/widgets/nt_widgets/nt_widget.dart'; import 'widget_container_model.dart'; class NTWidgetContainerModel extends WidgetContainerModel { + final NTConnection ntConnection; late NTWidget child; late NTWidgetModel childModel; NTWidgetContainerModel({ + required this.ntConnection, + required super.preferences, required super.initialPosition, required super.title, required this.childModel, @@ -26,7 +30,9 @@ class NTWidgetContainerModel extends WidgetContainerModel { }); NTWidgetContainerModel.fromJson({ + required this.ntConnection, required super.jsonData, + required super.preferences, super.enabled, super.onJsonLoadingWarning, }) : super.fromJson(); @@ -82,8 +88,13 @@ class NTWidgetContainerModel extends WidgetContainerModel { String type = tryCast(jsonData['type']) ?? ''; - childModel = NTWidgetBuilder.buildNTModelFromJson(type, widgetProperties, - onWidgetTypeNotFound: onJsonLoadingWarning); + childModel = NTWidgetBuilder.buildNTModelFromJson( + ntConnection, + preferences, + type, + widgetProperties, + onWidgetTypeNotFound: onJsonLoadingWarning, + ); } @override @@ -258,6 +269,8 @@ class NTWidgetContainerModel extends WidgetContainerModel { title: title, width: draggingRect.width, height: draggingRect.height, + cornerRadius: + preferences.getDouble(PrefKeys.cornerRadius) ?? Defaults.cornerRadius, opacity: 0.80, child: ChangeNotifierProvider.value( value: childModel, @@ -272,6 +285,8 @@ class NTWidgetContainerModel extends WidgetContainerModel { title: title, width: displayRect.width, height: displayRect.height, + cornerRadius: + preferences.getDouble(PrefKeys.cornerRadius) ?? Defaults.cornerRadius, opacity: (previewVisible) ? 0.25 : 1.00, child: Opacity( opacity: (enabled) ? 1.00 : 0.50, @@ -300,11 +315,15 @@ class NTWidgetContainerModel extends WidgetContainerModel { childModel.forceDispose(); childModel = NTWidgetBuilder.buildNTModelFromType( + ntConnection, + preferences, type, childModel.topic, dataType: childModel.dataType, - period: - (type != 'Graph') ? childModel.period : Settings.defaultGraphPeriod, + period: (type != 'Graph') + ? childModel.period + : preferences.getDouble(PrefKeys.defaultGraphPeriod) ?? + Defaults.defaultGraphPeriod, ); NTWidget? newWidget = NTWidgetBuilder.buildNTWidgetFromModel(childModel); diff --git a/lib/widgets/draggable_containers/models/widget_container_model.dart b/lib/widgets/draggable_containers/models/widget_container_model.dart index 31307287..4fa34bf0 100644 --- a/lib/widgets/draggable_containers/models/widget_container_model.dart +++ b/lib/widgets/draggable_containers/models/widget_container_model.dart @@ -2,6 +2,7 @@ import 'package:flutter/material.dart'; import 'package:dot_cast/dot_cast.dart'; import 'package:flutter_context_menu/flutter_context_menu.dart'; +import 'package:shared_preferences/shared_preferences.dart'; import 'package:elastic_dashboard/services/settings.dart'; import 'package:elastic_dashboard/widgets/dialog_widgets/dialog_text_input.dart'; @@ -9,23 +10,34 @@ import 'package:elastic_dashboard/widgets/draggable_containers/draggable_widget_ abstract class WidgetContainerModel extends ChangeNotifier { final Key key = UniqueKey(); + final SharedPreferences preferences; String? title; - bool draggable = !Settings.layoutLocked; + late bool draggable = + !(preferences.getBool(PrefKeys.layoutLocked) ?? Defaults.layoutLocked); bool _disposed = false; bool _forceDispose = false; - Rect draggingRect = Rect.fromLTWH( - 0, 0, Settings.gridSize.toDouble(), Settings.gridSize.toDouble()); + late Rect draggingRect = Rect.fromLTWH( + 0, + 0, + (preferences.getInt(PrefKeys.gridSize) ?? Defaults.gridSize).toDouble(), + (preferences.getInt(PrefKeys.gridSize) ?? Defaults.gridSize).toDouble()); Offset cursorGlobalLocation = const Offset(double.nan, double.nan); - Rect displayRect = Rect.fromLTWH( - 0, 0, Settings.gridSize.toDouble(), Settings.gridSize.toDouble()); + late Rect displayRect = Rect.fromLTWH( + 0, + 0, + (preferences.getInt(PrefKeys.gridSize) ?? Defaults.gridSize).toDouble(), + (preferences.getInt(PrefKeys.gridSize) ?? Defaults.gridSize).toDouble()); - Rect previewRect = Rect.fromLTWH( - 0, 0, Settings.gridSize.toDouble(), Settings.gridSize.toDouble()); + late Rect previewRect = Rect.fromLTWH( + 0, + 0, + (preferences.getInt(PrefKeys.gridSize) ?? Defaults.gridSize).toDouble(), + (preferences.getInt(PrefKeys.gridSize) ?? Defaults.gridSize).toDouble()); bool enabled = false; bool dragging = false; @@ -34,12 +46,15 @@ abstract class WidgetContainerModel extends ChangeNotifier { bool previewVisible = false; bool validLocation = true; - double minWidth = Settings.gridSize.toDouble(); - double minHeight = Settings.gridSize.toDouble(); + late double minWidth = + (preferences.getInt(PrefKeys.gridSize) ?? Defaults.gridSize).toDouble(); + late double minHeight = + (preferences.getInt(PrefKeys.gridSize) ?? Defaults.gridSize).toDouble(); late Rect dragStartLocation; WidgetContainerModel({ + required this.preferences, required Rect initialPosition, required this.title, this.enabled = false, @@ -52,6 +67,7 @@ abstract class WidgetContainerModel extends ChangeNotifier { WidgetContainerModel.fromJson({ required Map jsonData, + required this.preferences, this.enabled = false, this.minWidth = 128.0, this.minHeight = 128.0, @@ -106,9 +122,11 @@ abstract class WidgetContainerModel extends ChangeNotifier { double y = tryCast(jsonData['y']) ?? 0.0; - double width = tryCast(jsonData['width']) ?? Settings.gridSize.toDouble(); + double width = tryCast(jsonData['width']) ?? + (preferences.getInt(PrefKeys.gridSize) ?? Defaults.gridSize).toDouble(); - double height = tryCast(jsonData['height']) ?? Settings.gridSize.toDouble(); + double height = tryCast(jsonData['height']) ?? + (preferences.getInt(PrefKeys.gridSize) ?? Defaults.gridSize).toDouble(); displayRect = Rect.fromLTWH(x, y, width, height); } @@ -256,6 +274,8 @@ abstract class WidgetContainerModel extends ChangeNotifier { title: title, width: draggingRect.width, height: draggingRect.height, + cornerRadius: + preferences.getDouble(PrefKeys.cornerRadius) ?? Defaults.cornerRadius, opacity: 0.80, child: Container(), ); @@ -266,6 +286,8 @@ abstract class WidgetContainerModel extends ChangeNotifier { title: title, width: displayRect.width, height: displayRect.height, + cornerRadius: + preferences.getDouble(PrefKeys.cornerRadius) ?? Defaults.cornerRadius, child: Container(), ); } @@ -283,7 +305,9 @@ abstract class WidgetContainerModel extends ChangeNotifier { color: (validLocation) ? Colors.white.withOpacity(0.25) : Colors.black.withOpacity(0.1), - borderRadius: BorderRadius.circular(Settings.cornerRadius), + borderRadius: BorderRadius.circular( + preferences.getDouble(PrefKeys.cornerRadius) ?? + Defaults.cornerRadius), border: Border.all( color: (validLocation) ? Colors.lightGreenAccent.shade400 diff --git a/lib/widgets/editable_tab_bar.dart b/lib/widgets/editable_tab_bar.dart index 61df2898..5fa89342 100644 --- a/lib/widgets/editable_tab_bar.dart +++ b/lib/widgets/editable_tab_bar.dart @@ -3,37 +3,34 @@ import 'package:flutter/services.dart'; import 'package:flutter_context_menu/flutter_context_menu.dart'; import 'package:provider/provider.dart'; +import 'package:shared_preferences/shared_preferences.dart'; import 'package:transitioned_indexed_stack/transitioned_indexed_stack.dart'; import 'package:elastic_dashboard/services/settings.dart'; +import 'package:elastic_dashboard/util/tab_data.dart'; import 'package:elastic_dashboard/widgets/dialog_widgets/dialog_text_input.dart'; import 'package:elastic_dashboard/widgets/tab_grid.dart'; -class TabData { - String name; - - TabData({required this.name}); -} - class EditableTabBar extends StatelessWidget { - final List tabViews; + final SharedPreferences preferences; + final List tabData; - final Function(TabData tab) onTabCreate; + final Function() onTabCreate; final Function(int index) onTabDestroy; final Function() onTabMoveLeft; final Function() onTabMoveRight; final Function(int index, TabData newData) onTabRename; final Function(int index) onTabChanged; - final Function(int index, TabData newData) onTabDuplicate; + final Function(int index) onTabDuplicate; final int currentIndex; const EditableTabBar({ super.key, + required this.preferences, required this.currentIndex, required this.tabData, - required this.tabViews, required this.onTabCreate, required this.onTabDestroy, required this.onTabMoveLeft, @@ -77,17 +74,11 @@ class EditableTabBar extends StatelessWidget { } void duplicateTab(BuildContext context, int index) { - String tabName = '${tabData[index].name} (Copy)'; - TabData data = TabData(name: tabName); - - onTabDuplicate.call(index, data); + onTabDuplicate.call(index); } void createTab() { - String tabName = 'Tab ${tabData.length + 1}'; - TabData data = TabData(name: tabName); - - onTabCreate.call(data); + onTabCreate(); } void closeTab(int index) { @@ -133,7 +124,8 @@ class EditableTabBar extends StatelessWidget { onTabChanged.call(index); }, onSecondaryTapUp: (details) { - if (Settings.layoutLocked) { + if (preferences.getBool(PrefKeys.layoutLocked) ?? + Defaults.layoutLocked) { return; } ContextMenu contextMenu = ContextMenu( @@ -208,11 +200,15 @@ class EditableTabBar extends StatelessWidget { ), ), Visibility( - visible: !Settings.layoutLocked, + visible: !(preferences + .getBool(PrefKeys.layoutLocked) ?? + Defaults.layoutLocked), child: const SizedBox(width: 10), ), Visibility( - visible: !Settings.layoutLocked, + visible: !(preferences + .getBool(PrefKeys.layoutLocked) ?? + Defaults.layoutLocked), child: IconButton( onPressed: () { closeTab(index); @@ -243,7 +239,8 @@ class EditableTabBar extends StatelessWidget { children: [ IconButton( style: endButtonStyle, - onPressed: (!Settings.layoutLocked) + onPressed: !(preferences.getBool(PrefKeys.layoutLocked) ?? + Defaults.layoutLocked) ? () => onTabMoveLeft.call() : null, alignment: Alignment.center, @@ -251,14 +248,17 @@ class EditableTabBar extends StatelessWidget { ), IconButton( style: endButtonStyle, - onPressed: - (!Settings.layoutLocked) ? () => createTab() : null, + onPressed: !(preferences.getBool(PrefKeys.layoutLocked) ?? + Defaults.layoutLocked) + ? () => createTab() + : null, alignment: Alignment.center, icon: const Icon(Icons.add), ), IconButton( style: endButtonStyle, - onPressed: (!Settings.layoutLocked) + onPressed: !(preferences.getBool(PrefKeys.layoutLocked) ?? + Defaults.layoutLocked) ? () => onTabMoveRight.call() : null, alignment: Alignment.center, @@ -276,10 +276,13 @@ class EditableTabBar extends StatelessWidget { child: Stack( children: [ Visibility( - visible: Settings.showGrid, + visible: + preferences.getBool(PrefKeys.showGrid) ?? Defaults.showGrid, child: GridPaper( color: const Color.fromARGB(50, 195, 232, 243), - interval: Settings.gridSize.toDouble(), + interval: (preferences.getInt(PrefKeys.gridSize) ?? + Defaults.gridSize) + .toDouble(), divisions: 1, subdivisions: 1, child: Container(), @@ -289,10 +292,10 @@ class EditableTabBar extends StatelessWidget { curve: Curves.decelerate, index: currentIndex, children: [ - for (TabGrid grid in tabViews) - ChangeNotifierProvider( - create: (context) => TabGridModel(), - child: grid, + for (TabGridModel grid in tabData.map((e) => e.tabGrid)) + ChangeNotifierProvider.value( + value: grid, + child: const TabGrid(), ), ], ), diff --git a/lib/widgets/network_tree/networktables_tree.dart b/lib/widgets/network_tree/networktables_tree.dart index 05a7364b..ec16d970 100644 --- a/lib/widgets/network_tree/networktables_tree.dart +++ b/lib/widgets/network_tree/networktables_tree.dart @@ -4,6 +4,7 @@ import 'package:flutter/material.dart'; import 'package:collection/collection.dart'; import 'package:flutter_fancy_tree_view/flutter_fancy_tree_view.dart'; +import 'package:shared_preferences/shared_preferences.dart'; import 'package:elastic_dashboard/services/nt4_client.dart'; import 'package:elastic_dashboard/services/nt_connection.dart'; @@ -18,6 +19,8 @@ typedef ListLayoutBuilder = ListLayoutModel Function({ }); class NetworkTableTree extends StatefulWidget { + final NTConnection ntConnection; + final SharedPreferences preferences; final ListLayoutBuilder listLayoutBuilder; final Function(Offset globalPosition, WidgetContainerModel widget)? @@ -28,6 +31,8 @@ class NetworkTableTree extends StatefulWidget { const NetworkTableTree({ super.key, + required this.ntConnection, + required this.preferences, required this.listLayoutBuilder, required this.hideMetadata, this.onDragUpdate, @@ -39,7 +44,11 @@ class NetworkTableTree extends StatefulWidget { } class _NetworkTableTreeState extends State { - final NetworkTableTreeRow root = NetworkTableTreeRow(topic: '/', rowName: ''); + late final NetworkTableTreeRow root = NetworkTableTreeRow( + ntConnection: widget.ntConnection, + preferences: widget.preferences, + topic: '/', + rowName: ''); late final TreeController treeController; late final Function(Offset globalPosition, WidgetContainerModel widget)? @@ -65,8 +74,7 @@ class _NetworkTableTreeState extends State { }, ); - ntConnection.nt4Client - .addTopicAnnounceListener(onNewTopicAnnounced = (topic) { + widget.ntConnection.addTopicAnnounceListener(onNewTopicAnnounced = (topic) { setState(() { treeController.rebuild(); }); @@ -75,7 +83,7 @@ class _NetworkTableTreeState extends State { @override void dispose() { - ntConnection.nt4Client.removeTopicAnnounceListener(onNewTopicAnnounced); + widget.ntConnection.removeTopicAnnounceListener(onNewTopicAnnounced); super.dispose(); } @@ -126,7 +134,7 @@ class _NetworkTableTreeState extends State { Widget build(BuildContext context) { List topics = []; - for (NT4Topic topic in ntConnection.nt4Client.announcedTopics.values) { + for (NT4Topic topic in widget.ntConnection.announcedTopics().values) { if (topic.name == 'Time') { continue; } @@ -146,6 +154,7 @@ class _NetworkTableTreeState extends State { (BuildContext context, TreeEntry entry) { return TreeTile( key: UniqueKey(), + preferences: widget.preferences, entry: entry, listLayoutBuilder: widget.listLayoutBuilder, onDragUpdate: onDragUpdate, @@ -163,15 +172,7 @@ class _NetworkTableTreeState extends State { } class TreeTile extends StatelessWidget { - TreeTile({ - super.key, - required this.entry, - required this.onTap, - required this.listLayoutBuilder, - this.onDragUpdate, - this.onDragEnd, - }); - + final SharedPreferences preferences; final TreeEntry entry; final VoidCallback onTap; @@ -183,6 +184,16 @@ class TreeTile extends StatelessWidget { WidgetContainerModel? draggingWidget; + TreeTile({ + super.key, + required this.preferences, + required this.entry, + required this.onTap, + required this.listLayoutBuilder, + this.onDragUpdate, + this.onDragEnd, + }); + @override Widget build(BuildContext context) { TextStyle trailingStyle = diff --git a/lib/widgets/network_tree/networktables_tree_row.dart b/lib/widgets/network_tree/networktables_tree_row.dart index 706bfbe5..72ac7281 100644 --- a/lib/widgets/network_tree/networktables_tree_row.dart +++ b/lib/widgets/network_tree/networktables_tree_row.dart @@ -1,8 +1,11 @@ import 'package:flutter/material.dart'; +import 'package:shared_preferences/shared_preferences.dart'; + import 'package:elastic_dashboard/services/nt4_client.dart'; import 'package:elastic_dashboard/services/nt_connection.dart'; import 'package:elastic_dashboard/services/nt_widget_builder.dart'; +import 'package:elastic_dashboard/services/settings.dart'; import 'package:elastic_dashboard/widgets/draggable_containers/draggable_widget_container.dart'; import 'package:elastic_dashboard/widgets/draggable_containers/models/nt_widget_container_model.dart'; import 'package:elastic_dashboard/widgets/draggable_containers/models/widget_container_model.dart'; @@ -14,6 +17,8 @@ import 'package:elastic_dashboard/widgets/nt_widgets/single_topic/boolean_box.da import 'package:elastic_dashboard/widgets/nt_widgets/single_topic/text_display.dart'; class NetworkTableTreeRow { + final NTConnection ntConnection; + final SharedPreferences preferences; final String topic; final String rowName; @@ -22,6 +27,8 @@ class NetworkTableTreeRow { List children = []; NetworkTableTreeRow({ + required this.ntConnection, + required this.preferences, required this.topic, required this.rowName, this.ntTopic, @@ -74,8 +81,13 @@ class NetworkTableTreeRow { NetworkTableTreeRow createNewRow( {required String topic, required String name, NT4Topic? ntTopic}) { - NetworkTableTreeRow newRow = - NetworkTableTreeRow(topic: topic, rowName: name, ntTopic: ntTopic); + NetworkTableTreeRow newRow = NetworkTableTreeRow( + ntConnection: ntConnection, + preferences: preferences, + topic: topic, + rowName: name, + ntTopic: ntTopic, + ); addRow(newRow); return newRow; @@ -101,7 +113,8 @@ class NetworkTableTreeRow { children.clear(); } - static NTWidgetModel? getNTWidgetFromTopic(NT4Topic ntTopic) { + static NTWidgetModel? getNTWidgetFromTopic(NTConnection ntConnection, + SharedPreferences preferences, NT4Topic ntTopic) { switch (ntTopic.type) { case NT4TypeStr.kFloat64: case NT4TypeStr.kInt: @@ -113,11 +126,15 @@ class NetworkTableTreeRow { case NT4TypeStr.kString: case NT4TypeStr.kStringArr: return TextDisplayModel( + ntConnection: ntConnection, + preferences: preferences, topic: ntTopic.name, dataType: ntTopic.type, ); case NT4TypeStr.kBool: return BooleanBoxModel( + ntConnection: ntConnection, + preferences: preferences, topic: ntTopic.name, dataType: ntTopic.type, ); @@ -140,7 +157,8 @@ class NetworkTableTreeRow { (hasRow('description') || hasRow('connected')); if (isCameraStream) { - return CameraStreamModel(topic: topic); + return CameraStreamModel( + ntConnection: ntConnection, preferences: preferences, topic: topic); } if (hasRows([ @@ -152,13 +170,14 @@ class NetworkTableTreeRow { 'sizeFrontBack', 'sizeLeftRight', ])) { - return YAGSLSwerveDriveModel(topic: topic); + return YAGSLSwerveDriveModel( + ntConnection: ntConnection, preferences: preferences, topic: topic); } return null; } - return getNTWidgetFromTopic(ntTopic!); + return getNTWidgetFromTopic(ntConnection, preferences, ntTopic!); } Future getTypeString(String typeTopic) async { @@ -172,7 +191,8 @@ class NetworkTableTreeRow { return null; } - return NTWidgetBuilder.buildNTModelFromType(type, topic); + return NTWidgetBuilder.buildNTModelFromType( + ntConnection, preferences, type, topic); } Future?> getListLayoutChildren() async { @@ -227,6 +247,8 @@ class NetworkTableTreeRow { double height = NTWidgetBuilder.getDefaultHeight(primary); return NTWidgetContainerModel( + ntConnection: ntConnection, + preferences: preferences, initialPosition: Rect.fromLTWH(0.0, 0.0, width, height), title: rowName, childModel: primary, @@ -255,6 +277,8 @@ class NetworkTableTreeRow { title: rowName, width: width, height: height, + cornerRadius: + preferences.getDouble(PrefKeys.cornerRadius) ?? Defaults.cornerRadius, child: widget, ); } diff --git a/lib/widgets/nt_widgets/multi-topic/accelerometer.dart b/lib/widgets/nt_widgets/multi-topic/accelerometer.dart index 196907f8..081147ad 100644 --- a/lib/widgets/nt_widgets/multi-topic/accelerometer.dart +++ b/lib/widgets/nt_widgets/multi-topic/accelerometer.dart @@ -4,7 +4,6 @@ import 'package:dot_cast/dot_cast.dart'; import 'package:provider/provider.dart'; import 'package:elastic_dashboard/services/nt4_client.dart'; -import 'package:elastic_dashboard/services/nt_connection.dart'; import 'package:elastic_dashboard/widgets/nt_widgets/nt_widget.dart'; class AccelerometerModel extends NTWidgetModel { @@ -15,10 +14,19 @@ class AccelerometerModel extends NTWidgetModel { String get valueTopic => '$topic/Value'; - AccelerometerModel({required super.topic, super.dataType, super.period}) - : super(); + AccelerometerModel({ + required super.ntConnection, + required super.preferences, + required super.topic, + super.dataType, + super.period, + }) : super(); - AccelerometerModel.fromJson({required super.jsonData}) : super.fromJson(); + AccelerometerModel.fromJson({ + required super.ntConnection, + required super.preferences, + required super.jsonData, + }) : super.fromJson(); @override void init() { @@ -57,7 +65,7 @@ class AccelerometerWidget extends NTWidget { return StreamBuilder( stream: model.valueSubscription.periodicStream(yieldAll: false), - initialData: ntConnection.getLastAnnouncedValue(model.valueTopic), + initialData: model.ntConnection.getLastAnnouncedValue(model.valueTopic), builder: (context, snapshot) { double value = tryCast(snapshot.data) ?? 0.0; diff --git a/lib/widgets/nt_widgets/multi-topic/basic_swerve_drive.dart b/lib/widgets/nt_widgets/multi-topic/basic_swerve_drive.dart index 0e34b9a9..252648c8 100644 --- a/lib/widgets/nt_widgets/multi-topic/basic_swerve_drive.dart +++ b/lib/widgets/nt_widgets/multi-topic/basic_swerve_drive.dart @@ -6,7 +6,6 @@ import 'package:dot_cast/dot_cast.dart'; import 'package:provider/provider.dart'; import 'package:vector_math/vector_math_64.dart' show radians; -import 'package:elastic_dashboard/services/nt_connection.dart'; import 'package:elastic_dashboard/widgets/dialog_widgets/dialog_toggle_switch.dart'; import 'package:elastic_dashboard/widgets/nt_widgets/nt_widget.dart'; @@ -37,6 +36,8 @@ class BasicSwerveModel extends NTWidgetModel { String _rotationUnit = 'Radians'; BasicSwerveModel({ + required super.ntConnection, + required super.preferences, required super.topic, bool showRobotRotation = true, String rotationUnit = 'Radians', @@ -46,8 +47,11 @@ class BasicSwerveModel extends NTWidgetModel { _showRobotRotation = showRobotRotation, super(); - BasicSwerveModel.fromJson({required Map jsonData}) - : super.fromJson(jsonData: jsonData) { + BasicSwerveModel.fromJson({ + required super.ntConnection, + required super.preferences, + required Map jsonData, + }) : super.fromJson(jsonData: jsonData) { _showRobotRotation = tryCast(jsonData['show_robot_rotation']) ?? true; _rotationUnit = tryCast(jsonData['rotation_unit']) ?? 'Degrees'; } @@ -193,36 +197,36 @@ class SwerveDriveWidget extends NTWidget { return StreamBuilder( stream: model.multiTopicPeriodicStream, builder: (context, snapshot) { - double frontLeftAngle = tryCast(ntConnection + double frontLeftAngle = tryCast(model.ntConnection .getLastAnnouncedValue(model.frontLeftAngleTopic)) ?? 0.0; - double frontLeftVelocity = tryCast(ntConnection + double frontLeftVelocity = tryCast(model.ntConnection .getLastAnnouncedValue(model.frontLeftVelocityTopic)) ?? 0.0; - double frontRightAngle = tryCast(ntConnection + double frontRightAngle = tryCast(model.ntConnection .getLastAnnouncedValue(model.frontRightAngleTopic)) ?? 0.0; - double frontRightVelocity = tryCast(ntConnection + double frontRightVelocity = tryCast(model.ntConnection .getLastAnnouncedValue(model.frontRightVelocityTopic)) ?? 0.0; - double backLeftAngle = tryCast( - ntConnection.getLastAnnouncedValue(model.backLeftAngleTopic)) ?? + double backLeftAngle = tryCast(model.ntConnection + .getLastAnnouncedValue(model.backLeftAngleTopic)) ?? 0.0; - double backLeftVelocity = tryCast(ntConnection + double backLeftVelocity = tryCast(model.ntConnection .getLastAnnouncedValue(model.backLeftVelocityTopic)) ?? 0.0; - double backRightAngle = tryCast(ntConnection + double backRightAngle = tryCast(model.ntConnection .getLastAnnouncedValue(model.backRightAngleTopic)) ?? 0.0; - double backRightVelocity = tryCast(ntConnection + double backRightVelocity = tryCast(model.ntConnection .getLastAnnouncedValue(model.backRightVelocityTopic)) ?? 0.0; - double robotAngle = tryCast( - ntConnection.getLastAnnouncedValue(model.robotAngleTopic)) ?? + double robotAngle = tryCast(model.ntConnection + .getLastAnnouncedValue(model.robotAngleTopic)) ?? 0.0; if (model.rotationUnit == 'Degrees') { diff --git a/lib/widgets/nt_widgets/multi-topic/camera_stream.dart b/lib/widgets/nt_widgets/multi-topic/camera_stream.dart index f3130f60..1aec534a 100644 --- a/lib/widgets/nt_widgets/multi-topic/camera_stream.dart +++ b/lib/widgets/nt_widgets/multi-topic/camera_stream.dart @@ -4,7 +4,6 @@ import 'package:flutter/services.dart'; import 'package:dot_cast/dot_cast.dart'; import 'package:provider/provider.dart'; -import 'package:elastic_dashboard/services/nt_connection.dart'; import 'package:elastic_dashboard/widgets/custom_loading_indicator.dart'; import 'package:elastic_dashboard/widgets/dialog_widgets/dialog_text_input.dart'; import 'package:elastic_dashboard/widgets/mjpeg.dart'; @@ -60,6 +59,8 @@ class CameraStreamModel extends NTWidgetModel { } CameraStreamModel({ + required super.ntConnection, + required super.preferences, required super.topic, int? compression, int? fps, @@ -71,8 +72,11 @@ class CameraStreamModel extends NTWidgetModel { _resolution = resolution, super(); - CameraStreamModel.fromJson({required Map jsonData}) - : super.fromJson(jsonData: jsonData) { + CameraStreamModel.fromJson({ + required super.ntConnection, + required super.preferences, + required Map jsonData, + }) : super.fromJson(jsonData: jsonData) { _quality = tryCast(jsonData['compression']); _fps = tryCast(jsonData['fps']); @@ -257,9 +261,9 @@ class CameraStreamWidget extends NTWidget { return StreamBuilder( stream: model.multiTopicPeriodicStream, builder: (context, snapshot) { - List rawStreams = - tryCast(ntConnection.getLastAnnouncedValue(model.streamsTopic)) ?? - []; + List rawStreams = tryCast( + model.ntConnection.getLastAnnouncedValue(model.streamsTopic)) ?? + []; List streams = []; for (Object? stream in rawStreams) { @@ -272,7 +276,7 @@ class CameraStreamWidget extends NTWidget { streams.add(stream.substring('mjpg:'.length)); } - if (streams.isEmpty || !ntConnection.isNT4Connected) { + if (streams.isEmpty || !model.ntConnection.isNT4Connected) { return Stack( fit: StackFit.expand, children: [ @@ -292,9 +296,9 @@ class CameraStreamWidget extends NTWidget { CustomLoadingIndicator(), const SizedBox(height: 10), Text( - (ntConnection.isNT4Connected) + (model.ntConnection.isNT4Connected) ? 'Waiting for Camera Stream connection...' - : 'Waiting for Network Tables Connection...', + : 'Waiting for Network Tables connection...', textAlign: TextAlign.center, ), ], diff --git a/lib/widgets/nt_widgets/multi-topic/combo_box_chooser.dart b/lib/widgets/nt_widgets/multi-topic/combo_box_chooser.dart index 390a5219..7da578b6 100644 --- a/lib/widgets/nt_widgets/multi-topic/combo_box_chooser.dart +++ b/lib/widgets/nt_widgets/multi-topic/combo_box_chooser.dart @@ -5,7 +5,6 @@ import 'package:dropdown_button2/dropdown_button2.dart'; import 'package:provider/provider.dart'; import 'package:elastic_dashboard/services/nt4_client.dart'; -import 'package:elastic_dashboard/services/nt_connection.dart'; import 'package:elastic_dashboard/widgets/dialog_widgets/dialog_toggle_switch.dart'; import 'package:elastic_dashboard/widgets/nt_widgets/nt_widget.dart'; @@ -20,7 +19,14 @@ class ComboBoxChooserModel extends NTWidgetModel { final TextEditingController _searchController = TextEditingController(); - String? selectedChoice; + String? _selectedChoice; + + String? get selectedChoice => _selectedChoice; + + set selectedChoice(value) { + _selectedChoice = value; + refresh(); + } StringChooserData? previousData; @@ -37,6 +43,8 @@ class ComboBoxChooserModel extends NTWidgetModel { } ComboBoxChooserModel({ + required super.ntConnection, + required super.preferences, required super.topic, bool sortOptions = false, super.dataType, @@ -44,8 +52,11 @@ class ComboBoxChooserModel extends NTWidgetModel { }) : _sortOptions = sortOptions, super(); - ComboBoxChooserModel.fromJson({required Map jsonData}) - : super.fromJson(jsonData: jsonData) { + ComboBoxChooserModel.fromJson({ + required super.ntConnection, + required super.preferences, + required Map jsonData, + }) : super.fromJson(jsonData: jsonData) { _sortOptions = tryCast(jsonData['sort_options']) ?? _sortOptions; } @@ -82,8 +93,8 @@ class ComboBoxChooserModel extends NTWidgetModel { return; } - _selectedTopic ??= ntConnection.nt4Client - .publishNewTopic(selectedTopicName, NT4TypeStr.kString); + _selectedTopic ??= + ntConnection.publishNewTopic(selectedTopicName, NT4TypeStr.kString); ntConnection.updateDataFromTopic(_selectedTopic!, selected); } @@ -102,7 +113,7 @@ class ComboBoxChooserModel extends NTWidgetModel { } if (publishTopic) { - ntConnection.nt4Client.publishTopic(_activeTopic!); + ntConnection.publishTopic(_activeTopic!); } ntConnection.updateDataFromTopic(_activeTopic!, active); @@ -147,7 +158,7 @@ class ComboBoxChooser extends NTWidget { return StreamBuilder( stream: model.multiTopicPeriodicStream, builder: (context, snapshot) { - List rawOptions = ntConnection + List rawOptions = model.ntConnection .getLastAnnouncedValue(model.optionsTopicName) ?.tryCast>() ?? []; @@ -158,25 +169,25 @@ class ComboBoxChooser extends NTWidget { options.sort(); } - String? active = - tryCast(ntConnection.getLastAnnouncedValue(model.activeTopicName)); + String? active = tryCast( + model.ntConnection.getLastAnnouncedValue(model.activeTopicName)); if (active != null && active == '') { active = null; } String? selected = tryCast( - ntConnection.getLastAnnouncedValue(model.selectedTopicName)); + model.ntConnection.getLastAnnouncedValue(model.selectedTopicName)); if (selected != null && selected == '') { selected = null; } - String? defaultOption = - tryCast(ntConnection.getLastAnnouncedValue(model.defaultTopicName)); + String? defaultOption = tryCast( + model.ntConnection.getLastAnnouncedValue(model.defaultTopicName)); if (defaultOption != null && defaultOption == '') { defaultOption = null; } - if (!ntConnection.isNT4Connected) { + if (!model.ntConnection.isNT4Connected) { active = null; selected = null; defaultOption = null; diff --git a/lib/widgets/nt_widgets/multi-topic/command_scheduler.dart b/lib/widgets/nt_widgets/multi-topic/command_scheduler.dart index 6d1b7c5c..99e58bba 100644 --- a/lib/widgets/nt_widgets/multi-topic/command_scheduler.dart +++ b/lib/widgets/nt_widgets/multi-topic/command_scheduler.dart @@ -6,7 +6,6 @@ import 'package:dot_cast/dot_cast.dart'; import 'package:provider/provider.dart'; import 'package:elastic_dashboard/services/nt4_client.dart'; -import 'package:elastic_dashboard/services/nt_connection.dart'; import 'package:elastic_dashboard/widgets/nt_widgets/nt_widget.dart'; class CommandSchedulerModel extends NTWidgetModel { @@ -19,10 +18,19 @@ class CommandSchedulerModel extends NTWidgetModel { String get idsTopicName => '$topic/Ids'; String get cancelTopicName => '$topic/Cancel'; - CommandSchedulerModel({required super.topic, super.dataType, super.period}) - : super(); + CommandSchedulerModel({ + required super.ntConnection, + required super.preferences, + required super.topic, + super.dataType, + super.period, + }) : super(); - CommandSchedulerModel.fromJson({required super.jsonData}) : super.fromJson(); + CommandSchedulerModel.fromJson({ + required super.ntConnection, + required super.preferences, + required super.jsonData, + }) : super.fromJson(); @override void resetSubscription() { @@ -42,8 +50,8 @@ class CommandSchedulerModel extends NTWidgetModel { currentCancellations.add(id); - _cancelTopic ??= ntConnection.nt4Client - .publishNewTopic(cancelTopicName, NT4TypeStr.kIntArr); + _cancelTopic ??= + ntConnection.publishNewTopic(cancelTopicName, NT4TypeStr.kIntArr); if (_cancelTopic == null) { return; @@ -83,12 +91,12 @@ class CommandSchedulerWidget extends NTWidget { return StreamBuilder( stream: model.multiTopicPeriodicStream, builder: (context, snapshot) { - List rawNames = ntConnection + List rawNames = model.ntConnection .getLastAnnouncedValue(model.namesTopicName) ?.tryCast>() ?? []; - List rawIds = ntConnection + List rawIds = model.ntConnection .getLastAnnouncedValue(model.idsTopicName) ?.tryCast>() ?? []; diff --git a/lib/widgets/nt_widgets/multi-topic/command_widget.dart b/lib/widgets/nt_widgets/multi-topic/command_widget.dart index bc22acac..2937ff46 100644 --- a/lib/widgets/nt_widgets/multi-topic/command_widget.dart +++ b/lib/widgets/nt_widgets/multi-topic/command_widget.dart @@ -4,7 +4,6 @@ import 'package:dot_cast/dot_cast.dart'; import 'package:provider/provider.dart'; import 'package:elastic_dashboard/services/nt4_client.dart'; -import 'package:elastic_dashboard/services/nt_connection.dart'; import 'package:elastic_dashboard/widgets/dialog_widgets/dialog_toggle_switch.dart'; import 'package:elastic_dashboard/widgets/nt_widgets/nt_widget.dart'; @@ -27,6 +26,8 @@ class CommandModel extends NTWidgetModel { } CommandModel({ + required super.ntConnection, + required super.preferences, required super.topic, bool showType = true, super.dataType, @@ -34,8 +35,11 @@ class CommandModel extends NTWidgetModel { }) : _showType = showType, super(); - CommandModel.fromJson({required Map jsonData}) - : super.fromJson(jsonData: jsonData) { + CommandModel.fromJson({ + required super.ntConnection, + required super.preferences, + required Map jsonData, + }) : super.fromJson(jsonData: jsonData) { _showType = tryCast(jsonData['show_type']) ?? _showType; } @@ -92,11 +96,11 @@ class CommandWidget extends NTWidget { return StreamBuilder( stream: model.multiTopicPeriodicStream, builder: (context, snapshot) { - bool running = ntConnection + bool running = model.ntConnection .getLastAnnouncedValue(model.runningTopicName) ?.tryCast() ?? false; - String name = ntConnection + String name = model.ntConnection .getLastAnnouncedValue(model.nameTopicName) ?.tryCast() ?? 'Unknown'; @@ -120,17 +124,18 @@ class CommandWidget extends NTWidget { bool publishTopic = model.runningTopic == null; model.runningTopic = - ntConnection.getTopicFromName(model.runningTopicName); + model.ntConnection.getTopicFromName(model.runningTopicName); if (model.runningTopic == null) { return; } if (publishTopic) { - ntConnection.nt4Client.publishTopic(model.runningTopic!); + model.ntConnection.publishTopic(model.runningTopic!); } - ntConnection.updateDataFromTopic(model.runningTopic!, !running); + model.ntConnection + .updateDataFromTopic(model.runningTopic!, !running); }, child: AnimatedContainer( duration: const Duration(milliseconds: 50), diff --git a/lib/widgets/nt_widgets/multi-topic/differential_drive.dart b/lib/widgets/nt_widgets/multi-topic/differential_drive.dart index 9b79c51a..93a9ddbe 100644 --- a/lib/widgets/nt_widgets/multi-topic/differential_drive.dart +++ b/lib/widgets/nt_widgets/multi-topic/differential_drive.dart @@ -7,7 +7,6 @@ import 'package:provider/provider.dart'; import 'package:syncfusion_flutter_gauges/gauges.dart'; import 'package:elastic_dashboard/services/nt4_client.dart'; -import 'package:elastic_dashboard/services/nt_connection.dart'; import 'package:elastic_dashboard/widgets/nt_widgets/nt_widget.dart'; class DifferentialDriveModel extends NTWidgetModel { @@ -42,10 +41,19 @@ class DifferentialDriveModel extends NTWidgetModel { set rightSpeedCurrentValue(value) => _rightSpeedCurrentValue = value; - DifferentialDriveModel({required super.topic, super.dataType, super.period}) - : super(); + DifferentialDriveModel({ + required super.ntConnection, + required super.preferences, + required super.topic, + super.dataType, + super.period, + }) : super(); - DifferentialDriveModel.fromJson({required super.jsonData}) : super.fromJson(); + DifferentialDriveModel.fromJson({ + required super.ntConnection, + required super.preferences, + required super.jsonData, + }) : super.fromJson(); @override void resetSubscription() { @@ -91,10 +99,10 @@ class DifferentialDrive extends NTWidget { return StreamBuilder( stream: model.multiTopicPeriodicStream, builder: (context, snapshot) { - double leftSpeed = tryCast( - ntConnection.getLastAnnouncedValue(model.leftSpeedTopicName)) ?? + double leftSpeed = tryCast(model.ntConnection + .getLastAnnouncedValue(model.leftSpeedTopicName)) ?? 0.0; - double rightSpeed = tryCast(ntConnection + double rightSpeed = tryCast(model.ntConnection .getLastAnnouncedValue(model.rightSpeedTopicName)) ?? 0.0; @@ -134,19 +142,18 @@ class DifferentialDrive extends NTWidget { onChangeEnd: (value) { bool publishTopic = model.leftSpeedTopic == null; - model.leftSpeedTopic ??= - ntConnection.getTopicFromName(model.leftSpeedTopicName); + model.leftSpeedTopic ??= model.ntConnection + .getTopicFromName(model.leftSpeedTopicName); if (model.leftSpeedTopic == null) { return; } if (publishTopic) { - ntConnection.nt4Client - .publishTopic(model.leftSpeedTopic!); + model.ntConnection.publishTopic(model.leftSpeedTopic!); } - ntConnection.updateDataFromTopic( + model.ntConnection.updateDataFromTopic( model.leftSpeedTopic!, model.leftSpeedCurrentValue); model.leftSpeedPreviousValue = model.leftSpeedCurrentValue; @@ -207,7 +214,7 @@ class DifferentialDrive extends NTWidget { onChangeEnd: (value) { bool publishTopic = model.rightSpeedTopic == null; - model.rightSpeedTopic ??= ntConnection + model.rightSpeedTopic ??= model.ntConnection .getTopicFromName(model.rightSpeedTopicName); if (model.rightSpeedTopic == null) { @@ -215,11 +222,10 @@ class DifferentialDrive extends NTWidget { } if (publishTopic) { - ntConnection.nt4Client - .publishTopic(model.rightSpeedTopic!); + model.ntConnection.publishTopic(model.rightSpeedTopic!); } - ntConnection.updateDataFromTopic( + model.ntConnection.updateDataFromTopic( model.rightSpeedTopic!, model.rightSpeedCurrentValue); model.rightSpeedPreviousValue = diff --git a/lib/widgets/nt_widgets/multi-topic/encoder_widget.dart b/lib/widgets/nt_widgets/multi-topic/encoder_widget.dart index 7322ac25..2d0f39be 100644 --- a/lib/widgets/nt_widgets/multi-topic/encoder_widget.dart +++ b/lib/widgets/nt_widgets/multi-topic/encoder_widget.dart @@ -3,7 +3,6 @@ import 'package:flutter/material.dart'; import 'package:dot_cast/dot_cast.dart'; import 'package:provider/provider.dart'; -import 'package:elastic_dashboard/services/nt_connection.dart'; import 'package:elastic_dashboard/widgets/nt_widgets/nt_widget.dart'; class EncoderModel extends NTWidgetModel { @@ -13,9 +12,19 @@ class EncoderModel extends NTWidgetModel { String get distanceTopic => '$topic/Distance'; String get speedTopic => '$topic/Speed'; - EncoderModel({required super.topic, super.dataType, super.period}) : super(); + EncoderModel({ + required super.ntConnection, + required super.preferences, + required super.topic, + super.dataType, + super.period, + }) : super(); - EncoderModel.fromJson({required super.jsonData}) : super.fromJson(); + EncoderModel.fromJson({ + required super.ntConnection, + required super.preferences, + required super.jsonData, + }) : super.fromJson(); @override List getCurrentData() { @@ -40,12 +49,12 @@ class EncoderWidget extends NTWidget { return StreamBuilder( stream: model.multiTopicPeriodicStream, builder: (context, snapshot) { - double distance = - tryCast(ntConnection.getLastAnnouncedValue(model.distanceTopic)) ?? - 0.0; - double speed = - tryCast(ntConnection.getLastAnnouncedValue(model.speedTopic)) ?? - 0.0; + double distance = tryCast(model.ntConnection + .getLastAnnouncedValue(model.distanceTopic)) ?? + 0.0; + double speed = tryCast( + model.ntConnection.getLastAnnouncedValue(model.speedTopic)) ?? + 0.0; return Column( mainAxisAlignment: MainAxisAlignment.spaceEvenly, diff --git a/lib/widgets/nt_widgets/multi-topic/field_widget.dart b/lib/widgets/nt_widgets/multi-topic/field_widget.dart index fa4a27ca..080c0319 100644 --- a/lib/widgets/nt_widgets/multi-topic/field_widget.dart +++ b/lib/widgets/nt_widgets/multi-topic/field_widget.dart @@ -10,7 +10,6 @@ import 'package:vector_math/vector_math_64.dart' show radians; import 'package:elastic_dashboard/services/field_images.dart'; import 'package:elastic_dashboard/services/nt4_client.dart'; -import 'package:elastic_dashboard/services/nt_connection.dart'; import 'package:elastic_dashboard/services/settings.dart'; import 'package:elastic_dashboard/services/text_formatter_builder.dart'; import 'package:elastic_dashboard/widgets/dialog_widgets/dialog_dropdown_chooser.dart'; @@ -45,6 +44,8 @@ class FieldWidgetModel extends NTWidgetModel { late Function(NT4Topic topic) topicAnnounceListener; FieldWidgetModel({ + required super.ntConnection, + required super.preferences, required super.topic, String? fieldName, bool showOtherObjects = true, @@ -63,8 +64,11 @@ class FieldWidgetModel extends NTWidgetModel { _field = FieldImages.getFieldFromGame(_fieldGame)!; } - FieldWidgetModel.fromJson({required Map jsonData}) - : super.fromJson(jsonData: jsonData) { + FieldWidgetModel.fromJson({ + required super.ntConnection, + required super.preferences, + required Map jsonData, + }) : super.fromJson(jsonData: jsonData) { _fieldGame = tryCast(jsonData['field_game']) ?? _fieldGame; _robotWidthMeters = tryCast(jsonData['robot_width']) ?? 0.85; @@ -97,7 +101,7 @@ class FieldWidgetModel extends NTWidgetModel { } }; - ntConnection.nt4Client.addTopicAnnounceListener(topicAnnounceListener); + ntConnection.addTopicAnnounceListener(topicAnnounceListener); } @override @@ -114,7 +118,7 @@ class FieldWidgetModel extends NTWidgetModel { if (deleting) { _field.dispose(); - ntConnection.nt4Client.removeTopicAnnounceListener(topicAnnounceListener); + ntConnection.removeTopicAnnounceListener(topicAnnounceListener); } _widgetSize = null; @@ -318,7 +322,8 @@ class FieldWidgetModel extends NTWidgetModel { Stream get multiTopicPeriodicStream async* { final Duration delayTime = Duration( microseconds: ((subscription?.options.periodicRateSeconds ?? - Settings.defaultPeriod) * + preferences.getDouble(PrefKeys.defaultPeriod) ?? + Defaults.defaultPeriod) * 1e6) .round()); @@ -497,7 +502,7 @@ class FieldWidget extends NTWidget { return StreamBuilder( stream: model.multiTopicPeriodicStream, builder: (context, snapshot) { - List robotPositionRaw = ntConnection + List robotPositionRaw = model.ntConnection .getLastAnnouncedValue(model._robotTopicName) ?.tryCast>() ?? []; @@ -553,7 +558,7 @@ class FieldWidget extends NTWidget { if (model.showOtherObjects || model.showTrajectories) { for (String objectTopic in model._otherObjectTopics) { - List? objectPositionRaw = ntConnection + List? objectPositionRaw = model.ntConnection .getLastAnnouncedValue(objectTopic) ?.tryCast>(); diff --git a/lib/widgets/nt_widgets/multi-topic/fms_info.dart b/lib/widgets/nt_widgets/multi-topic/fms_info.dart index 92026e6b..52a9e076 100644 --- a/lib/widgets/nt_widgets/multi-topic/fms_info.dart +++ b/lib/widgets/nt_widgets/multi-topic/fms_info.dart @@ -4,7 +4,6 @@ import 'package:dot_cast/dot_cast.dart'; import 'package:patterns_canvas/patterns_canvas.dart'; import 'package:provider/provider.dart'; -import 'package:elastic_dashboard/services/nt_connection.dart'; import 'package:elastic_dashboard/widgets/nt_widgets/nt_widget.dart'; class FMSInfoModel extends NTWidgetModel { @@ -19,9 +18,19 @@ class FMSInfoModel extends NTWidgetModel { String get replayNumberTopic => '$topic/ReplayNumber'; String get stationNumberTopic => '$topic/StationNumber'; - FMSInfoModel({required super.topic, super.dataType, super.period}); + FMSInfoModel({ + required super.ntConnection, + required super.preferences, + required super.topic, + super.dataType, + super.period, + }); - FMSInfoModel.fromJson({required super.jsonData}) : super.fromJson(); + FMSInfoModel.fromJson({ + required super.ntConnection, + required super.preferences, + required super.jsonData, + }) : super.fromJson(); @override List getCurrentData() { @@ -85,23 +94,23 @@ class FMSInfo extends NTWidget { return StreamBuilder( stream: model.multiTopicPeriodicStream, builder: (context, snapshot) { - String eventName = - tryCast(ntConnection.getLastAnnouncedValue(model.eventNameTopic)) ?? - ''; - int controlData = tryCast( - ntConnection.getLastAnnouncedValue(model.controlDataTopic)) ?? + String eventName = tryCast(model.ntConnection + .getLastAnnouncedValue(model.eventNameTopic)) ?? + ''; + int controlData = tryCast(model.ntConnection + .getLastAnnouncedValue(model.controlDataTopic)) ?? 32; - bool redAlliance = - tryCast(ntConnection.getLastAnnouncedValue(model.allianceTopic)) ?? - true; - int matchNumber = tryCast( - ntConnection.getLastAnnouncedValue(model.matchNumberTopic)) ?? + bool redAlliance = tryCast(model.ntConnection + .getLastAnnouncedValue(model.allianceTopic)) ?? + true; + int matchNumber = tryCast(model.ntConnection + .getLastAnnouncedValue(model.matchNumberTopic)) ?? 0; - int matchType = - tryCast(ntConnection.getLastAnnouncedValue(model.matchTypeTopic)) ?? - 0; - int replayNumber = tryCast( - ntConnection.getLastAnnouncedValue(model.replayNumberTopic)) ?? + int matchType = tryCast(model.ntConnection + .getLastAnnouncedValue(model.matchTypeTopic)) ?? + 0; + int replayNumber = tryCast(model.ntConnection + .getLastAnnouncedValue(model.replayNumberTopic)) ?? 0; String eventNameDisplay = '$eventName${(eventName != '') ? ' ' : ''}'; diff --git a/lib/widgets/nt_widgets/multi-topic/gyro.dart b/lib/widgets/nt_widgets/multi-topic/gyro.dart index a18aee09..c93ec51b 100644 --- a/lib/widgets/nt_widgets/multi-topic/gyro.dart +++ b/lib/widgets/nt_widgets/multi-topic/gyro.dart @@ -5,7 +5,6 @@ import 'package:provider/provider.dart'; import 'package:syncfusion_flutter_gauges/gauges.dart'; import 'package:elastic_dashboard/services/nt4_client.dart'; -import 'package:elastic_dashboard/services/nt_connection.dart'; import 'package:elastic_dashboard/widgets/dialog_widgets/dialog_toggle_switch.dart'; import 'package:elastic_dashboard/widgets/nt_widgets/nt_widget.dart'; @@ -26,16 +25,21 @@ class GyroModel extends NTWidgetModel { refresh(); } - GyroModel( - {required super.topic, - bool counterClockwisePositive = false, - super.dataType, - super.period}) - : _counterClockwisePositive = counterClockwisePositive, + GyroModel({ + required super.ntConnection, + required super.preferences, + required super.topic, + bool counterClockwisePositive = false, + super.dataType, + super.period, + }) : _counterClockwisePositive = counterClockwisePositive, super(); - GyroModel.fromJson({required Map jsonData}) - : super.fromJson(jsonData: jsonData) { + GyroModel.fromJson({ + required super.ntConnection, + required super.preferences, + required Map jsonData, + }) : super.fromJson(jsonData: jsonData) { _counterClockwisePositive = tryCast(jsonData['counter_clockwise_positive']) ?? false; } @@ -106,7 +110,7 @@ class Gyro extends NTWidget { return StreamBuilder( stream: model.valueSubscription.periodicStream(yieldAll: false), - initialData: ntConnection.getLastAnnouncedValue(model.valueTopic), + initialData: model.ntConnection.getLastAnnouncedValue(model.valueTopic), builder: (context, snapshot) { double value = tryCast(snapshot.data) ?? 0.0; diff --git a/lib/widgets/nt_widgets/multi-topic/motor_controller.dart b/lib/widgets/nt_widgets/multi-topic/motor_controller.dart index 322de437..91205635 100644 --- a/lib/widgets/nt_widgets/multi-topic/motor_controller.dart +++ b/lib/widgets/nt_widgets/multi-topic/motor_controller.dart @@ -5,7 +5,6 @@ import 'package:provider/provider.dart'; import 'package:syncfusion_flutter_gauges/gauges.dart'; import 'package:elastic_dashboard/services/nt4_client.dart'; -import 'package:elastic_dashboard/services/nt_connection.dart'; import 'package:elastic_dashboard/widgets/nt_widgets/nt_widget.dart'; class MotorControllerModel extends NTWidgetModel { @@ -16,10 +15,19 @@ class MotorControllerModel extends NTWidgetModel { late NT4Subscription valueSubscription; - MotorControllerModel({required super.topic, super.dataType, super.period}) - : super(); + MotorControllerModel({ + required super.ntConnection, + required super.preferences, + required super.topic, + super.dataType, + super.period, + }) : super(); - MotorControllerModel.fromJson({required super.jsonData}) : super.fromJson(); + MotorControllerModel.fromJson({ + required super.ntConnection, + required super.preferences, + required super.jsonData, + }) : super.fromJson(); @override void init() { @@ -56,7 +64,7 @@ class MotorController extends NTWidget { return StreamBuilder( stream: model.valueSubscription.periodicStream(yieldAll: false), - initialData: ntConnection.getLastAnnouncedValue(model.valueTopic), + initialData: model.ntConnection.getLastAnnouncedValue(model.valueTopic), builder: (context, snapshot) { double value = tryCast(snapshot.data) ?? 0.0; diff --git a/lib/widgets/nt_widgets/multi-topic/network_alerts.dart b/lib/widgets/nt_widgets/multi-topic/network_alerts.dart index c8f09dea..b6501804 100644 --- a/lib/widgets/nt_widgets/multi-topic/network_alerts.dart +++ b/lib/widgets/nt_widgets/multi-topic/network_alerts.dart @@ -3,7 +3,6 @@ import 'package:flutter/material.dart'; import 'package:dot_cast/dot_cast.dart'; import 'package:provider/provider.dart'; -import 'package:elastic_dashboard/services/nt_connection.dart'; import 'package:elastic_dashboard/widgets/nt_widgets/nt_widget.dart'; class NetworkAlertsModel extends NTWidgetModel { @@ -14,10 +13,19 @@ class NetworkAlertsModel extends NTWidgetModel { String get warningsTopicName => '$topic/warnings'; String get infosTopicName => '$topic/infos'; - NetworkAlertsModel({required super.topic, super.dataType, super.period}) - : super(); + NetworkAlertsModel({ + required super.ntConnection, + required super.preferences, + required super.topic, + super.dataType, + super.period, + }) : super(); - NetworkAlertsModel.fromJson({required super.jsonData}) : super.fromJson(); + NetworkAlertsModel.fromJson({ + required super.ntConnection, + required super.preferences, + required super.jsonData, + }) : super.fromJson(); @override List getCurrentData() { @@ -56,17 +64,17 @@ class NetworkAlerts extends NTWidget { return StreamBuilder( stream: model.multiTopicPeriodicStream, builder: (context, snapshot) { - List errorsRaw = ntConnection + List errorsRaw = model.ntConnection .getLastAnnouncedValue(model.errorsTopicName) ?.tryCast>() ?? []; - List warningsRaw = ntConnection + List warningsRaw = model.ntConnection .getLastAnnouncedValue(model.warningsTopicName) ?.tryCast>() ?? []; - List infosRaw = ntConnection + List infosRaw = model.ntConnection .getLastAnnouncedValue(model.infosTopicName) ?.tryCast>() ?? []; diff --git a/lib/widgets/nt_widgets/multi-topic/pid_controller.dart b/lib/widgets/nt_widgets/multi-topic/pid_controller.dart index 05981712..d0b91abd 100644 --- a/lib/widgets/nt_widgets/multi-topic/pid_controller.dart +++ b/lib/widgets/nt_widgets/multi-topic/pid_controller.dart @@ -4,7 +4,6 @@ import 'package:dot_cast/dot_cast.dart'; import 'package:provider/provider.dart'; import 'package:elastic_dashboard/services/nt4_client.dart'; -import 'package:elastic_dashboard/services/nt_connection.dart'; import 'package:elastic_dashboard/services/text_formatter_builder.dart'; import 'package:elastic_dashboard/widgets/dialog_widgets/dialog_text_input.dart'; import 'package:elastic_dashboard/widgets/nt_widgets/nt_widget.dart'; @@ -49,10 +48,19 @@ class PIDControllerModel extends NTWidgetModel { set setpointLastValue(value) => _setpointLastValue = value; - PIDControllerModel({required super.topic, super.dataType, super.period}) - : super(); + PIDControllerModel({ + required super.ntConnection, + required super.preferences, + required super.topic, + super.dataType, + super.period, + }) : super(); - PIDControllerModel.fromJson({required super.jsonData}) : super.fromJson(); + PIDControllerModel.fromJson({ + required super.ntConnection, + required super.preferences, + required super.jsonData, + }) : super.fromJson(); @override void resetSubscription() { @@ -76,7 +84,7 @@ class PIDControllerModel extends NTWidgetModel { } if (publishTopic) { - ntConnection.nt4Client.publishTopic(_kpTopic!); + ntConnection.publishTopic(_kpTopic!); } ntConnection.updateDataFromTopic(_kpTopic!, data); @@ -94,7 +102,7 @@ class PIDControllerModel extends NTWidgetModel { } if (publishTopic) { - ntConnection.nt4Client.publishTopic(_kiTopic!); + ntConnection.publishTopic(_kiTopic!); } ntConnection.updateDataFromTopic(_kiTopic!, data); @@ -112,7 +120,7 @@ class PIDControllerModel extends NTWidgetModel { } if (publishTopic) { - ntConnection.nt4Client.publishTopic(_kdTopic!); + ntConnection.publishTopic(_kdTopic!); } ntConnection.updateDataFromTopic(_kdTopic!, data); @@ -130,7 +138,7 @@ class PIDControllerModel extends NTWidgetModel { } if (publishTopic) { - ntConnection.nt4Client.publishTopic(_setpointTopic!); + ntConnection.publishTopic(_setpointTopic!); } ntConnection.updateDataFromTopic(_setpointTopic!, data); @@ -173,16 +181,16 @@ class PIDControllerWidget extends NTWidget { return StreamBuilder( stream: model.multiTopicPeriodicStream, builder: (context, snapshot) { - double kP = - tryCast(ntConnection.getLastAnnouncedValue(model.kpTopicName)) ?? - 0.0; - double kI = - tryCast(ntConnection.getLastAnnouncedValue(model.kiTopicName)) ?? - 0.0; - double kD = - tryCast(ntConnection.getLastAnnouncedValue(model.kdTopicName)) ?? - 0.0; - double setpoint = tryCast(ntConnection + double kP = tryCast(model.ntConnection + .getLastAnnouncedValue(model.kpTopicName)) ?? + 0.0; + double kI = tryCast(model.ntConnection + .getLastAnnouncedValue(model.kiTopicName)) ?? + 0.0; + double kD = tryCast(model.ntConnection + .getLastAnnouncedValue(model.kdTopicName)) ?? + 0.0; + double setpoint = tryCast(model.ntConnection .getLastAnnouncedValue(model.setpointTopicName)) ?? 0.0; diff --git a/lib/widgets/nt_widgets/multi-topic/power_distribution.dart b/lib/widgets/nt_widgets/multi-topic/power_distribution.dart index a264b7ca..0d090758 100644 --- a/lib/widgets/nt_widgets/multi-topic/power_distribution.dart +++ b/lib/widgets/nt_widgets/multi-topic/power_distribution.dart @@ -3,7 +3,6 @@ import 'package:flutter/material.dart'; import 'package:dot_cast/dot_cast.dart'; import 'package:provider/provider.dart'; -import 'package:elastic_dashboard/services/nt_connection.dart'; import 'package:elastic_dashboard/widgets/nt_widgets/nt_widget.dart'; class PowerDistributionModel extends NTWidgetModel { @@ -17,10 +16,19 @@ class PowerDistributionModel extends NTWidgetModel { String get voltageTopic => '$topic/Voltage'; String get currentTopic => '$topic/TotalCurrent'; - PowerDistributionModel({required super.topic, super.dataType, super.period}) - : super(); + PowerDistributionModel({ + required super.ntConnection, + required super.preferences, + required super.topic, + super.dataType, + super.period, + }) : super(); - PowerDistributionModel.fromJson({required super.jsonData}) : super.fromJson(); + PowerDistributionModel.fromJson({ + required super.ntConnection, + required super.preferences, + required super.jsonData, + }) : super.fromJson(); @override void init() { @@ -71,7 +79,7 @@ class PowerDistribution extends NTWidget { List channels = []; for (int channel = start; channel <= end; channel++) { - double current = tryCast(ntConnection + double current = tryCast(model.ntConnection .getLastAnnouncedValue(model.channelTopics[channel])) ?? 0.0; @@ -109,7 +117,7 @@ class PowerDistribution extends NTWidget { List channels = []; for (int channel = start; channel >= end; channel--) { - double current = tryCast(ntConnection + double current = tryCast(model.ntConnection .getLastAnnouncedValue(model.channelTopics[channel])) ?? 0.0; @@ -150,12 +158,12 @@ class PowerDistribution extends NTWidget { return StreamBuilder( stream: model.multiTopicPeriodicStream, builder: (context, snapshot) { - double voltage = - tryCast(ntConnection.getLastAnnouncedValue(model.voltageTopic)) ?? - 0.0; - double totalCurrent = - tryCast(ntConnection.getLastAnnouncedValue(model.currentTopic)) ?? - 0.0; + double voltage = tryCast( + model.ntConnection.getLastAnnouncedValue(model.voltageTopic)) ?? + 0.0; + double totalCurrent = tryCast( + model.ntConnection.getLastAnnouncedValue(model.currentTopic)) ?? + 0.0; return Column( children: [ diff --git a/lib/widgets/nt_widgets/multi-topic/profiled_pid_controller.dart b/lib/widgets/nt_widgets/multi-topic/profiled_pid_controller.dart index 25b5c599..bc7643fc 100644 --- a/lib/widgets/nt_widgets/multi-topic/profiled_pid_controller.dart +++ b/lib/widgets/nt_widgets/multi-topic/profiled_pid_controller.dart @@ -4,7 +4,6 @@ import 'package:dot_cast/dot_cast.dart'; import 'package:provider/provider.dart'; import 'package:elastic_dashboard/services/nt4_client.dart'; -import 'package:elastic_dashboard/services/nt_connection.dart'; import 'package:elastic_dashboard/services/text_formatter_builder.dart'; import 'package:elastic_dashboard/widgets/dialog_widgets/dialog_text_input.dart'; import 'package:elastic_dashboard/widgets/nt_widgets/nt_widget.dart'; @@ -49,12 +48,19 @@ class ProfiledPIDControllerModel extends NTWidgetModel { set goalLastValue(value) => _goalLastValue = value; - ProfiledPIDControllerModel( - {required super.topic, super.dataType, super.period}) - : super(); + ProfiledPIDControllerModel({ + required super.ntConnection, + required super.preferences, + required super.topic, + super.dataType, + super.period, + }) : super(); - ProfiledPIDControllerModel.fromJson({required super.jsonData}) - : super.fromJson(); + ProfiledPIDControllerModel.fromJson({ + required super.ntConnection, + required super.preferences, + required super.jsonData, + }) : super.fromJson(); @override void resetSubscription() { @@ -78,7 +84,7 @@ class ProfiledPIDControllerModel extends NTWidgetModel { } if (publishTopic) { - ntConnection.nt4Client.publishTopic(_kpTopic!); + ntConnection.publishTopic(_kpTopic!); } ntConnection.updateDataFromTopic(_kpTopic!, data); @@ -96,7 +102,7 @@ class ProfiledPIDControllerModel extends NTWidgetModel { } if (publishTopic) { - ntConnection.nt4Client.publishTopic(_kiTopic!); + ntConnection.publishTopic(_kiTopic!); } ntConnection.updateDataFromTopic(_kiTopic!, data); @@ -114,7 +120,7 @@ class ProfiledPIDControllerModel extends NTWidgetModel { } if (publishTopic) { - ntConnection.nt4Client.publishTopic(_kdTopic!); + ntConnection.publishTopic(_kdTopic!); } ntConnection.updateDataFromTopic(_kdTopic!, data); @@ -132,7 +138,7 @@ class ProfiledPIDControllerModel extends NTWidgetModel { } if (publishTopic) { - ntConnection.nt4Client.publishTopic(_goalTopic!); + ntConnection.publishTopic(_goalTopic!); } ntConnection.updateDataFromTopic(_goalTopic!, data); @@ -175,17 +181,17 @@ class ProfiledPIDControllerWidget extends NTWidget { return StreamBuilder( stream: model.multiTopicPeriodicStream, builder: (context, snapshot) { - double kP = - tryCast(ntConnection.getLastAnnouncedValue(model.kpTopicName)) ?? - 0.0; - double kI = - tryCast(ntConnection.getLastAnnouncedValue(model.kiTopicName)) ?? - 0.0; - double kD = - tryCast(ntConnection.getLastAnnouncedValue(model.kdTopicName)) ?? - 0.0; - double goal = tryCast( - ntConnection.getLastAnnouncedValue(model.goalTopicName)) ?? + double kP = tryCast(model.ntConnection + .getLastAnnouncedValue(model.kpTopicName)) ?? + 0.0; + double kI = tryCast(model.ntConnection + .getLastAnnouncedValue(model.kiTopicName)) ?? + 0.0; + double kD = tryCast(model.ntConnection + .getLastAnnouncedValue(model.kdTopicName)) ?? + 0.0; + double goal = tryCast(model.ntConnection + .getLastAnnouncedValue(model.goalTopicName)) ?? 0.0; // Creates the text editing controllers if they are null diff --git a/lib/widgets/nt_widgets/multi-topic/relay_widget.dart b/lib/widgets/nt_widgets/multi-topic/relay_widget.dart index 96529e91..76b80172 100644 --- a/lib/widgets/nt_widgets/multi-topic/relay_widget.dart +++ b/lib/widgets/nt_widgets/multi-topic/relay_widget.dart @@ -4,7 +4,6 @@ import 'package:dot_cast/dot_cast.dart'; import 'package:provider/provider.dart'; import 'package:elastic_dashboard/services/nt4_client.dart'; -import 'package:elastic_dashboard/services/nt_connection.dart'; import 'package:elastic_dashboard/widgets/nt_widgets/nt_widget.dart'; class RelayModel extends NTWidgetModel { @@ -18,9 +17,19 @@ class RelayModel extends NTWidgetModel { final List selectedOptions = ['Off', 'On', 'Forward', 'Reverse']; - RelayModel({required super.topic, super.dataType, super.period}) : super(); + RelayModel({ + required super.ntConnection, + required super.preferences, + required super.topic, + super.dataType, + super.period, + }) : super(); - RelayModel.fromJson({required super.jsonData}) : super.fromJson(); + RelayModel.fromJson({ + required super.ntConnection, + required super.preferences, + required super.jsonData, + }) : super.fromJson(); @override void init() { @@ -59,7 +68,8 @@ class RelayWidget extends NTWidget { return StreamBuilder( stream: model.valueSubscription.periodicStream(yieldAll: false), - initialData: ntConnection.getLastAnnouncedValue(model.valueTopicName), + initialData: + model.ntConnection.getLastAnnouncedValue(model.valueTopicName), builder: (context, snapshot) { String selected = tryCast(snapshot.data) ?? 'Off'; @@ -84,18 +94,19 @@ class RelayWidget extends NTWidget { bool publishTopic = model.valueTopic == null; - model.valueTopic ??= - ntConnection.getTopicFromName(model.valueTopicName); + model.valueTopic ??= model.ntConnection + .getTopicFromName(model.valueTopicName); if (model.valueTopic == null) { return; } if (publishTopic) { - ntConnection.nt4Client.publishTopic(model.valueTopic!); + model.ntConnection.publishTopic(model.valueTopic!); } - ntConnection.updateDataFromTopic(model.valueTopic!, option); + model.ntConnection + .updateDataFromTopic(model.valueTopic!, option); }, isSelected: model.selectedOptions.map((e) => selected == e).toList(), diff --git a/lib/widgets/nt_widgets/multi-topic/robot_preferences.dart b/lib/widgets/nt_widgets/multi-topic/robot_preferences.dart index 9d188ec5..ab8c49b4 100644 --- a/lib/widgets/nt_widgets/multi-topic/robot_preferences.dart +++ b/lib/widgets/nt_widgets/multi-topic/robot_preferences.dart @@ -5,7 +5,6 @@ import 'package:provider/provider.dart'; import 'package:searchable_listview/searchable_listview.dart'; import 'package:elastic_dashboard/services/nt4_client.dart'; -import 'package:elastic_dashboard/services/nt_connection.dart'; import 'package:elastic_dashboard/widgets/nt_widgets/nt_widget.dart'; class RobotPreferencesModel extends NTWidgetModel { @@ -21,10 +20,19 @@ class RobotPreferencesModel extends NTWidgetModel { PreferenceSearch? searchWidget; - RobotPreferencesModel({required super.topic, super.dataType, super.period}) - : super(); - - RobotPreferencesModel.fromJson({required super.jsonData}) : super.fromJson(); + RobotPreferencesModel({ + required super.ntConnection, + required super.preferences, + required super.topic, + super.dataType, + super.period, + }) : super(); + + RobotPreferencesModel.fromJson({ + required super.ntConnection, + required super.preferences, + required super.jsonData, + }) : super.fromJson(); } class RobotPreferences extends NTWidget { @@ -41,8 +49,7 @@ class RobotPreferences extends NTWidget { builder: (context, snapshot) { bool rebuildWidget = model.searchWidget == null; - for (NT4Topic nt4Topic - in ntConnection.nt4Client.announcedTopics.values) { + for (NT4Topic nt4Topic in model.ntConnection.announcedTopics().values) { if (!nt4Topic.name.contains(model.topic) || model.preferenceTopicNames.contains(nt4Topic.name) || nt4Topic.name.contains('.type')) { @@ -50,7 +57,7 @@ class RobotPreferences extends NTWidget { } Object? previousValue = - ntConnection.getLastAnnouncedValue(nt4Topic.name); + model.ntConnection.getLastAnnouncedValue(nt4Topic.name); model.preferenceTopicNames.add(nt4Topic.name); model.preferenceTopics.addAll({nt4Topic.name: nt4Topic}); @@ -64,9 +71,9 @@ class RobotPreferences extends NTWidget { } Iterable announcedTopics = - ntConnection.nt4Client.announcedTopics.values.map( - (e) => e.name, - ); + model.ntConnection.announcedTopics().values.map( + (e) => e.name, + ); for (String topic in model.preferenceTopicNames) { if (!announcedTopics.contains(topic)) { @@ -81,13 +88,13 @@ class RobotPreferences extends NTWidget { continue; } - if (ntConnection.getLastAnnouncedValue(topic).toString() != + if (model.ntConnection.getLastAnnouncedValue(topic).toString() != model.previousValues[topic].toString()) { model.preferenceTextControllers[topic]?.text = - ntConnection.getLastAnnouncedValue(topic).toString(); + model.ntConnection.getLastAnnouncedValue(topic).toString(); model.previousValues[topic] = - ntConnection.getLastAnnouncedValue(topic); + model.ntConnection.getLastAnnouncedValue(topic); } } @@ -134,9 +141,9 @@ class RobotPreferences extends NTWidget { return; } - ntConnection.nt4Client.publishTopic(nt4Topic); - ntConnection.updateDataFromTopic(nt4Topic, formattedData); - ntConnection.nt4Client.unpublishTopic(nt4Topic); + model.ntConnection.publishTopic(nt4Topic); + model.ntConnection.updateDataFromTopic(nt4Topic, formattedData); + model.ntConnection.unpublishTopic(nt4Topic); model.preferenceTextControllers[topic]?.text = formattedData.toString(); @@ -189,8 +196,11 @@ class PreferenceSearch extends StatelessWidget { spaceBetweenSearchAndList: 15, filter: (query) { return preferenceTopicNames - .where((element) => - element.toLowerCase().contains(query.toLowerCase())) + .where((element) => element + .split('/') + .last + .toLowerCase() + .contains(query.toLowerCase())) .toList(); }, initialList: preferenceTopicNames, diff --git a/lib/widgets/nt_widgets/multi-topic/split_button_chooser.dart b/lib/widgets/nt_widgets/multi-topic/split_button_chooser.dart index 9446b066..229ab3a0 100644 --- a/lib/widgets/nt_widgets/multi-topic/split_button_chooser.dart +++ b/lib/widgets/nt_widgets/multi-topic/split_button_chooser.dart @@ -4,7 +4,6 @@ import 'package:dot_cast/dot_cast.dart'; import 'package:provider/provider.dart'; import 'package:elastic_dashboard/services/nt4_client.dart'; -import 'package:elastic_dashboard/services/nt_connection.dart'; import 'package:elastic_dashboard/widgets/nt_widgets/multi-topic/combo_box_chooser.dart'; import 'package:elastic_dashboard/widgets/nt_widgets/nt_widget.dart'; @@ -25,13 +24,18 @@ class SplitButtonChooserModel extends NTWidgetModel { NT4Topic? _activeTopic; SplitButtonChooserModel({ + required super.ntConnection, + required super.preferences, required super.topic, super.dataType, super.period, }) : super(); - SplitButtonChooserModel.fromJson({required super.jsonData}) - : super.fromJson(); + SplitButtonChooserModel.fromJson({ + required super.ntConnection, + required super.preferences, + required super.jsonData, + }) : super.fromJson(); @override void resetSubscription() { @@ -45,8 +49,8 @@ class SplitButtonChooserModel extends NTWidgetModel { return; } - _selectedTopic ??= ntConnection.nt4Client - .publishNewTopic(selectedTopicName, NT4TypeStr.kString); + _selectedTopic ??= + ntConnection.publishNewTopic(selectedTopicName, NT4TypeStr.kString); ntConnection.updateDataFromTopic(_selectedTopic!, selected); } @@ -65,7 +69,7 @@ class SplitButtonChooserModel extends NTWidgetModel { } if (publishTopic) { - ntConnection.nt4Client.publishTopic(_activeTopic!); + ntConnection.publishTopic(_activeTopic!); } ntConnection.updateDataFromTopic(_activeTopic!, active); @@ -105,32 +109,32 @@ class SplitButtonChooser extends NTWidget { return StreamBuilder( stream: model.multiTopicPeriodicStream, builder: (context, snapshot) { - List rawOptions = ntConnection + List rawOptions = model.ntConnection .getLastAnnouncedValue(model.optionsTopicName) ?.tryCast>() ?? []; List options = rawOptions.whereType().toList(); - String? active = - tryCast(ntConnection.getLastAnnouncedValue(model.activeTopicName)); + String? active = tryCast( + model.ntConnection.getLastAnnouncedValue(model.activeTopicName)); if (active != null && active == '') { active = null; } String? selected = tryCast( - ntConnection.getLastAnnouncedValue(model.selectedTopicName)); + model.ntConnection.getLastAnnouncedValue(model.selectedTopicName)); if (selected != null && selected == '') { selected = null; } - String? defaultOption = - tryCast(ntConnection.getLastAnnouncedValue(model.defaultTopicName)); + String? defaultOption = tryCast( + model.ntConnection.getLastAnnouncedValue(model.defaultTopicName)); if (defaultOption != null && defaultOption == '') { defaultOption = null; } - if (!ntConnection.isNT4Connected) { + if (!model.ntConnection.isNT4Connected) { active = null; selected = null; defaultOption = null; diff --git a/lib/widgets/nt_widgets/multi-topic/subsystem_widget.dart b/lib/widgets/nt_widgets/multi-topic/subsystem_widget.dart index b3b420c9..ba6cca45 100644 --- a/lib/widgets/nt_widgets/multi-topic/subsystem_widget.dart +++ b/lib/widgets/nt_widgets/multi-topic/subsystem_widget.dart @@ -3,7 +3,6 @@ import 'package:flutter/material.dart'; import 'package:dot_cast/dot_cast.dart'; import 'package:provider/provider.dart'; -import 'package:elastic_dashboard/services/nt_connection.dart'; import 'package:elastic_dashboard/widgets/nt_widgets/nt_widget.dart'; class SubsystemModel extends NTWidgetModel { @@ -13,10 +12,19 @@ class SubsystemModel extends NTWidgetModel { String get defaultCommandTopic => '$topic/.default'; String get currentCommandTopic => '$topic/.command'; - SubsystemModel({required super.topic, super.dataType, super.period}) - : super(); + SubsystemModel({ + required super.ntConnection, + required super.preferences, + required super.topic, + super.dataType, + super.period, + }) : super(); - SubsystemModel.fromJson({required super.jsonData}) : super.fromJson(); + SubsystemModel.fromJson({ + required super.ntConnection, + required super.preferences, + required super.jsonData, + }) : super.fromJson(); @override List getCurrentData() { @@ -43,10 +51,10 @@ class SubsystemWidget extends NTWidget { return StreamBuilder( stream: model.multiTopicPeriodicStream, builder: (context, snapshot) { - String defaultCommand = tryCast(ntConnection + String defaultCommand = tryCast(model.ntConnection .getLastAnnouncedValue(model.defaultCommandTopic)) ?? 'none'; - String currentCommand = tryCast(ntConnection + String currentCommand = tryCast(model.ntConnection .getLastAnnouncedValue(model.currentCommandTopic)) ?? 'none'; diff --git a/lib/widgets/nt_widgets/multi-topic/three_axis_accelerometer.dart b/lib/widgets/nt_widgets/multi-topic/three_axis_accelerometer.dart index 4c0bd293..27e60076 100644 --- a/lib/widgets/nt_widgets/multi-topic/three_axis_accelerometer.dart +++ b/lib/widgets/nt_widgets/multi-topic/three_axis_accelerometer.dart @@ -3,7 +3,6 @@ import 'package:flutter/material.dart'; import 'package:dot_cast/dot_cast.dart'; import 'package:provider/provider.dart'; -import 'package:elastic_dashboard/services/nt_connection.dart'; import 'package:elastic_dashboard/widgets/nt_widgets/nt_widget.dart'; class ThreeAxisAccelerometerModel extends NTWidgetModel { @@ -14,12 +13,19 @@ class ThreeAxisAccelerometerModel extends NTWidgetModel { String get yTopic => '$topic/Y'; String get zTopic => '$topic/Z'; - ThreeAxisAccelerometerModel( - {required super.topic, super.dataType, super.period}) - : super(); + ThreeAxisAccelerometerModel({ + required super.ntConnection, + required super.preferences, + required super.topic, + super.dataType, + super.period, + }) : super(); - ThreeAxisAccelerometerModel.fromJson({required super.jsonData}) - : super.fromJson(); + ThreeAxisAccelerometerModel.fromJson({ + required super.ntConnection, + required super.preferences, + required super.jsonData, + }) : super.fromJson(); @override List getCurrentData() { @@ -44,11 +50,14 @@ class ThreeAxisAccelerometer extends NTWidget { stream: model.multiTopicPeriodicStream, builder: (context, snapshot) { double xAccel = - tryCast(ntConnection.getLastAnnouncedValue(model.xTopic)) ?? 0.0; + tryCast(model.ntConnection.getLastAnnouncedValue(model.xTopic)) ?? + 0.0; double yAccel = - tryCast(ntConnection.getLastAnnouncedValue(model.yTopic)) ?? 0.0; + tryCast(model.ntConnection.getLastAnnouncedValue(model.yTopic)) ?? + 0.0; double zAccel = - tryCast(ntConnection.getLastAnnouncedValue(model.zTopic)) ?? 0.0; + tryCast(model.ntConnection.getLastAnnouncedValue(model.zTopic)) ?? + 0.0; return Column( mainAxisAlignment: MainAxisAlignment.spaceEvenly, diff --git a/lib/widgets/nt_widgets/multi-topic/ultrasonic.dart b/lib/widgets/nt_widgets/multi-topic/ultrasonic.dart index 8075a821..70bfc321 100644 --- a/lib/widgets/nt_widgets/multi-topic/ultrasonic.dart +++ b/lib/widgets/nt_widgets/multi-topic/ultrasonic.dart @@ -4,7 +4,6 @@ import 'package:dot_cast/dot_cast.dart'; import 'package:provider/provider.dart'; import 'package:elastic_dashboard/services/nt4_client.dart'; -import 'package:elastic_dashboard/services/nt_connection.dart'; import 'package:elastic_dashboard/widgets/nt_widgets/nt_widget.dart'; class UltrasonicModel extends NTWidgetModel { @@ -15,10 +14,19 @@ class UltrasonicModel extends NTWidgetModel { late NT4Subscription valueSubscription; - UltrasonicModel({required super.topic, super.dataType, super.period}) - : super(); + UltrasonicModel({ + required super.ntConnection, + required super.preferences, + required super.topic, + super.dataType, + super.period, + }) : super(); - UltrasonicModel.fromJson({required super.jsonData}) : super.fromJson(); + UltrasonicModel.fromJson({ + required super.ntConnection, + required super.preferences, + required super.jsonData, + }) : super.fromJson(); @override void init() { @@ -55,7 +63,7 @@ class Ultrasonic extends NTWidget { return StreamBuilder( stream: model.valueSubscription.periodicStream(yieldAll: false), - initialData: ntConnection.getLastAnnouncedValue(model.valueTopic), + initialData: model.ntConnection.getLastAnnouncedValue(model.valueTopic), builder: (context, snapshot) { double value = tryCast(snapshot.data) ?? 0.0; diff --git a/lib/widgets/nt_widgets/multi-topic/yagsl_swerve_drive.dart b/lib/widgets/nt_widgets/multi-topic/yagsl_swerve_drive.dart index 4e52780d..2d75e283 100644 --- a/lib/widgets/nt_widgets/multi-topic/yagsl_swerve_drive.dart +++ b/lib/widgets/nt_widgets/multi-topic/yagsl_swerve_drive.dart @@ -5,7 +5,6 @@ import 'package:flutter/material.dart'; import 'package:dot_cast/dot_cast.dart'; import 'package:provider/provider.dart'; import 'package:vector_math/vector_math_64.dart' show radians; -import 'package:elastic_dashboard/services/nt_connection.dart'; import 'package:elastic_dashboard/widgets/dialog_widgets/dialog_toggle_switch.dart'; import 'package:elastic_dashboard/widgets/nt_widgets/nt_widget.dart'; @@ -18,6 +17,8 @@ class YAGSLSwerveDriveModel extends NTWidgetModel { bool _showDesiredStates; YAGSLSwerveDriveModel({ + required super.ntConnection, + required super.preferences, required super.topic, bool showRobotRotation = true, bool showDesiredStates = true, @@ -27,30 +28,13 @@ class YAGSLSwerveDriveModel extends NTWidgetModel { _showRobotRotation = showRobotRotation, super(); - YAGSLSwerveDriveModel.fromJson({required Map jsonData}) - : _showRobotRotation = tryCast(jsonData['show_robot_rotation']) ?? true, - _showDesiredStates = tryCast(jsonData['show_desired_states']) ?? true, - _angleOffset = tryCast(jsonData['angle_offset']) ?? 0.0, - super.fromJson(jsonData: jsonData); - - String get measuredStatesTopic => '$topic/measuredStates'; - String get desiredStatesTopic => '$topic/desiredStates'; - String get robotRotationTopic => '$topic/robotRotation'; - String get maxSpeedTopic => '$topic/maxSpeed'; - String get robotWidthTopic => '$topic/sizeLeftRight'; - String get robotLengthTopic => '$topic/sizeFrontBack'; - String get rotationUnitTopic => '$topic/rotationUnit'; - - bool get showRobotRotation => _showRobotRotation; - set showRobotRotation(bool value) { - _showRobotRotation = value; - refresh(); - } - - bool get showDesiredStates => _showDesiredStates; - set showDesiredStates(bool value) { - _showDesiredStates = value; - refresh(); + YAGSLSwerveDriveModel.fromJson({ + required super.ntConnection, + required super.preferences, + required Map jsonData, + }) : super.fromJson(jsonData: jsonData) { + _showRobotRotation = tryCast(jsonData['show_robot_rotation']) ?? true; + _showDesiredStates = tryCast(jsonData['show_desired_states']) ?? true; } @override @@ -155,17 +139,15 @@ class YAGSLSwerveDrive extends NTWidget { Widget build(BuildContext context) { YAGSLSwerveDriveModel model = cast(context.watch()); - return GestureDetector( - behavior: HitTestBehavior.translucent, - child: StreamBuilder( - stream: model.multiTopicPeriodicStream, - builder: (context, snapshot) { - List rawMeasuredStates = tryCast(ntConnection - .getLastAnnouncedValue(model.measuredStatesTopic)) ?? - []; - List rawDesiredStates = tryCast(ntConnection - .getLastAnnouncedValue(model.desiredStatesTopic)) ?? - []; + return StreamBuilder( + stream: model.multiTopicPeriodicStream, + builder: (context, snapshot) { + List measuredStatesRaw = tryCast(model.ntConnection + .getLastAnnouncedValue(model.measuredStatesTopic)) ?? + []; + List desiredStatesRaw = tryCast(model.ntConnection + .getLastAnnouncedValue(model.desiredStatesTopic)) ?? + []; // Filter and cast to List List measuredStates = @@ -173,12 +155,12 @@ class YAGSLSwerveDrive extends NTWidget { List desiredStates = List.from(rawDesiredStates.whereType()); - double width = tryCast( - ntConnection.getLastAnnouncedValue(model.robotWidthTopic)) ?? - 1.0; - double length = tryCast( - ntConnection.getLastAnnouncedValue(model.robotLengthTopic)) ?? - width; + double width = tryCast(model.ntConnection + .getLastAnnouncedValue(model.robotWidthTopic)) ?? + 1.0; + double length = tryCast(model.ntConnection + .getLastAnnouncedValue(model.robotLengthTopic)) ?? + width; width = width > 0.0 ? width : 1.0; length = length > 0.0 ? length : 0.0; @@ -186,12 +168,13 @@ class YAGSLSwerveDrive extends NTWidget { double sizeRatio = min(length, width) / max(length, width); double lengthWidthRatio = length / width; - String rotationUnit = tryCast(ntConnection - .getLastAnnouncedValue(model.rotationUnitTopic)) ?? - 'radians'; - double robotAngle = tryCast(ntConnection - .getLastAnnouncedValue(model.robotRotationTopic)) ?? - 0.0; + String rotationUnit = tryCast(model.ntConnection + .getLastAnnouncedValue(model.rotationUnitTopic)) ?? + 'radians'; + + double robotAngle = tryCast(model.ntConnection + .getLastAnnouncedValue(model.robotRotationTopic)) ?? + 0.0; if (rotationUnit == 'degrees') { robotAngle = radians(robotAngle + model._angleOffset); @@ -199,10 +182,13 @@ class YAGSLSwerveDrive extends NTWidget { robotAngle *= 2 * pi + model._angleOffset; } - double maxSpeed = tryCast( - ntConnection.getLastAnnouncedValue(model.maxSpeedTopic)) ?? - 4.5; - maxSpeed = maxSpeed > 0.0 ? maxSpeed : 4.5; + double maxSpeed = tryCast(model.ntConnection + .getLastAnnouncedValue(model.maxSpeedTopic)) ?? + 4.5; + + if (maxSpeed <= 0.0) { + maxSpeed = 4.5; + } return LayoutBuilder( builder: (context, constraints) { diff --git a/lib/widgets/nt_widgets/nt_widget.dart b/lib/widgets/nt_widgets/nt_widget.dart index 9a81563d..8e814320 100644 --- a/lib/widgets/nt_widgets/nt_widget.dart +++ b/lib/widgets/nt_widgets/nt_widget.dart @@ -2,6 +2,7 @@ import 'package:flutter/material.dart'; import 'package:collection/collection.dart'; import 'package:dot_cast/dot_cast.dart'; +import 'package:shared_preferences/shared_preferences.dart'; import 'package:elastic_dashboard/services/nt4_client.dart'; import 'package:elastic_dashboard/services/nt_connection.dart'; @@ -20,6 +21,9 @@ import 'package:elastic_dashboard/widgets/nt_widgets/single_topic/toggle_switch. import 'package:elastic_dashboard/widgets/nt_widgets/single_topic/voltage_view.dart'; class NTWidgetModel extends ChangeNotifier { + final NTConnection ntConnection; + final SharedPreferences preferences; + String _typeOverride = 'NTWidget'; String get type => _typeOverride; @@ -42,30 +46,44 @@ class NTWidgetModel extends ChangeNotifier { bool _forceDispose = false; NTWidgetModel({ + required this.ntConnection, + required this.preferences, required String topic, this.dataType = 'Unknown', double? period, }) : _topic = topic { - this.period = period ?? Settings.defaultPeriod; + this.period = period ?? + preferences.getDouble(PrefKeys.defaultPeriod) ?? + Defaults.defaultPeriod; init(); } NTWidgetModel.createDefault({ + required this.ntConnection, + required this.preferences, required String type, required String topic, this.dataType = 'Unknown', double? period, }) : _typeOverride = type, _topic = topic { - this.period = period ?? Settings.defaultPeriod; + this.period = period ?? + preferences.getDouble(PrefKeys.defaultPeriod) ?? + Defaults.defaultPeriod; init(); } - NTWidgetModel.fromJson({required Map jsonData}) { + NTWidgetModel.fromJson({ + required this.ntConnection, + required this.preferences, + required Map jsonData, + }) { _topic = tryCast(jsonData['topic']) ?? ''; - _period = tryCast(jsonData['period']) ?? Settings.defaultPeriod; + _period = tryCast(jsonData['period']) ?? + preferences.getDouble(PrefKeys.defaultPeriod) ?? + Defaults.defaultPeriod; dataType = tryCast(jsonData['data_type']) ?? dataType; init(); @@ -195,7 +213,8 @@ class NTWidgetModel extends ChangeNotifier { Stream get multiTopicPeriodicStream async* { final Duration delayTime = Duration( microseconds: ((subscription?.options.periodicRateSeconds ?? - Settings.defaultPeriod) * + preferences.getDouble(PrefKeys.defaultPeriod) ?? + Defaults.defaultPeriod) * 1e6) .round()); diff --git a/lib/widgets/nt_widgets/single_topic/boolean_box.dart b/lib/widgets/nt_widgets/single_topic/boolean_box.dart index 528b8ab1..207a9f7a 100644 --- a/lib/widgets/nt_widgets/single_topic/boolean_box.dart +++ b/lib/widgets/nt_widgets/single_topic/boolean_box.dart @@ -3,7 +3,6 @@ import 'package:flutter/material.dart'; import 'package:dot_cast/dot_cast.dart'; import 'package:provider/provider.dart'; -import 'package:elastic_dashboard/services/nt_connection.dart'; import 'package:elastic_dashboard/widgets/dialog_widgets/dialog_color_picker.dart'; import 'package:elastic_dashboard/widgets/dialog_widgets/dialog_dropdown_chooser.dart'; import 'package:elastic_dashboard/widgets/nt_widgets/nt_widget.dart'; @@ -56,22 +55,27 @@ class BooleanBoxModel extends NTWidgetModel { refresh(); } - BooleanBoxModel( - {required super.topic, - Color trueColor = Colors.green, - Color falseColor = Colors.red, - String trueIcon = 'None', - String falseIcon = 'None', - super.dataType, - super.period}) - : _falseColor = falseColor, + BooleanBoxModel({ + required super.ntConnection, + required super.preferences, + required super.topic, + Color trueColor = Colors.green, + Color falseColor = Colors.red, + String trueIcon = 'None', + String falseIcon = 'None', + super.dataType, + super.period, + }) : _falseColor = falseColor, _trueColor = trueColor, _trueIcon = trueIcon, _falseIcon = falseIcon, super(); - BooleanBoxModel.fromJson({required Map jsonData}) - : super.fromJson(jsonData: jsonData) { + BooleanBoxModel.fromJson({ + required super.ntConnection, + required super.preferences, + required Map jsonData, + }) : super.fromJson(jsonData: jsonData) { int? trueColorValue = tryCast(jsonData['true_color']) ?? tryCast(jsonData['colorWhenTrue']); int? falseColorValue = @@ -200,7 +204,7 @@ class BooleanBox extends NTWidget { return StreamBuilder( stream: model.subscription?.periodicStream(yieldAll: false), - initialData: ntConnection.getLastAnnouncedValue(model.topic), + initialData: model.ntConnection.getLastAnnouncedValue(model.topic), builder: (context, snapshot) { bool value = tryCast(snapshot.data) ?? false; diff --git a/lib/widgets/nt_widgets/single_topic/graph.dart b/lib/widgets/nt_widgets/single_topic/graph.dart index 13eabdd0..ac350ab3 100644 --- a/lib/widgets/nt_widgets/single_topic/graph.dart +++ b/lib/widgets/nt_widgets/single_topic/graph.dart @@ -61,6 +61,8 @@ class GraphModel extends NTWidgetModel { _GraphWidgetGraph? _graphWidget; GraphModel({ + required super.ntConnection, + required super.preferences, required super.topic, double timeDisplayed = 5.0, double? minValue, @@ -76,8 +78,11 @@ class GraphModel extends NTWidgetModel { _lineWidth = lineWidth, super(); - GraphModel.fromJson({required Map jsonData}) - : super.fromJson(jsonData: jsonData) { + GraphModel.fromJson({ + required super.ntConnection, + required super.preferences, + required Map jsonData, + }) : super.fromJson(jsonData: jsonData) { _timeDisplayed = tryCast(jsonData['time_displayed']) ?? tryCast(jsonData['visibleTime']) ?? 5.0; @@ -92,8 +97,8 @@ class GraphModel extends NTWidgetModel { return { ...super.toJson(), 'time_displayed': _timeDisplayed, - 'min_value': _minValue, - 'max_value': _maxValue, + if (_minValue != null) 'min_value': _minValue, + if (_maxValue != null) 'max_value': _maxValue, 'color': _mainColor.value, 'line_width': _lineWidth, }; diff --git a/lib/widgets/nt_widgets/single_topic/match_time.dart b/lib/widgets/nt_widgets/single_topic/match_time.dart index 40ecb8de..4df92a37 100644 --- a/lib/widgets/nt_widgets/single_topic/match_time.dart +++ b/lib/widgets/nt_widgets/single_topic/match_time.dart @@ -4,7 +4,6 @@ import 'package:flutter/services.dart'; import 'package:dot_cast/dot_cast.dart'; import 'package:provider/provider.dart'; -import 'package:elastic_dashboard/services/nt_connection.dart'; import 'package:elastic_dashboard/widgets/dialog_widgets/dialog_dropdown_chooser.dart'; import 'package:elastic_dashboard/widgets/dialog_widgets/dialog_text_input.dart'; import 'package:elastic_dashboard/widgets/nt_widgets/nt_widget.dart'; @@ -44,20 +43,25 @@ class MatchTimeModel extends NTWidgetModel { refresh(); } - MatchTimeModel( - {required super.topic, - String timeDisplayMode = 'Minutes and Seconds', - int redStartTime = 15, - int yellowStartTime = 30, - super.dataType, - super.period}) - : _timeDisplayMode = timeDisplayMode, + MatchTimeModel({ + required super.ntConnection, + required super.preferences, + required super.topic, + String timeDisplayMode = 'Minutes and Seconds', + int redStartTime = 15, + int yellowStartTime = 30, + super.dataType, + super.period, + }) : _timeDisplayMode = timeDisplayMode, _yellowStartTime = yellowStartTime, _redStartTime = redStartTime, super(); - MatchTimeModel.fromJson({required Map jsonData}) - : super.fromJson(jsonData: jsonData) { + MatchTimeModel.fromJson({ + required super.ntConnection, + required super.preferences, + required Map jsonData, + }) : super.fromJson(jsonData: jsonData) { _timeDisplayMode = tryCast(jsonData['time_display_mode']) ?? 'Minutes and Seconds'; @@ -178,7 +182,7 @@ class MatchTimeWidget extends NTWidget { return StreamBuilder( stream: model.subscription?.periodicStream(yieldAll: false), - initialData: ntConnection.getLastAnnouncedValue(model.topic), + initialData: model.ntConnection.getLastAnnouncedValue(model.topic), builder: (context, snapshot) { double time = tryCast(snapshot.data) ?? -1.0; time = time.floorToDouble(); diff --git a/lib/widgets/nt_widgets/single_topic/multi_color_view.dart b/lib/widgets/nt_widgets/single_topic/multi_color_view.dart index 6f33d23d..8c428a2d 100644 --- a/lib/widgets/nt_widgets/single_topic/multi_color_view.dart +++ b/lib/widgets/nt_widgets/single_topic/multi_color_view.dart @@ -3,7 +3,6 @@ import 'package:flutter/material.dart'; import 'package:dot_cast/dot_cast.dart'; import 'package:provider/provider.dart'; -import 'package:elastic_dashboard/services/nt_connection.dart'; import 'package:elastic_dashboard/widgets/nt_widgets/nt_widget.dart'; class MultiColorView extends NTWidget { @@ -17,7 +16,7 @@ class MultiColorView extends NTWidget { return StreamBuilder( stream: model.subscription?.periodicStream(yieldAll: false), - initialData: ntConnection.getLastAnnouncedValue(model.topic), + initialData: model.ntConnection.getLastAnnouncedValue(model.topic), builder: (context, snapshot) { List hexStringsRaw = snapshot.data?.tryCast>() ?? []; diff --git a/lib/widgets/nt_widgets/single_topic/number_bar.dart b/lib/widgets/nt_widgets/single_topic/number_bar.dart index 1d8002ec..6fea9230 100644 --- a/lib/widgets/nt_widgets/single_topic/number_bar.dart +++ b/lib/widgets/nt_widgets/single_topic/number_bar.dart @@ -5,7 +5,6 @@ import 'package:dot_cast/dot_cast.dart'; import 'package:provider/provider.dart'; import 'package:syncfusion_flutter_gauges/gauges.dart'; -import 'package:elastic_dashboard/services/nt_connection.dart'; import 'package:elastic_dashboard/services/text_formatter_builder.dart'; import 'package:elastic_dashboard/widgets/dialog_widgets/dialog_dropdown_chooser.dart'; import 'package:elastic_dashboard/widgets/dialog_widgets/dialog_text_input.dart'; @@ -58,6 +57,8 @@ class NumberBarModel extends NTWidgetModel { } NumberBarModel({ + required super.ntConnection, + required super.preferences, required super.topic, double minValue = -1.0, double maxValue = 1.0, @@ -73,8 +74,11 @@ class NumberBarModel extends NTWidgetModel { _minValue = minValue, super(); - NumberBarModel.fromJson({required Map jsonData}) - : super.fromJson(jsonData: jsonData) { + NumberBarModel.fromJson({ + required super.ntConnection, + required super.preferences, + required Map jsonData, + }) : super.fromJson(jsonData: jsonData) { _minValue = tryCast(jsonData['min_value']) ?? -1.0; _maxValue = tryCast(jsonData['max_value']) ?? 1.0; _divisions = tryCast(jsonData['divisions']); @@ -86,11 +90,11 @@ class NumberBarModel extends NTWidgetModel { Map toJson() { return { ...super.toJson(), - 'min_value': _minValue, - 'max_value': _maxValue, - 'divisions': _divisions, - 'inverted': _inverted, - 'orientation': _orientation, + 'min_value': minValue, + 'max_value': maxValue, + if (divisions != null) 'divisions': divisions, + 'inverted': inverted, + 'orientation': orientation, }; } @@ -203,7 +207,7 @@ class NumberBar extends NTWidget { return StreamBuilder( stream: model.subscription?.periodicStream(yieldAll: false), - initialData: ntConnection.getLastAnnouncedValue(model.topic), + initialData: model.ntConnection.getLastAnnouncedValue(model.topic), builder: (context, snapshot) { double value = tryCast(snapshot.data) ?? 0.0; diff --git a/lib/widgets/nt_widgets/single_topic/number_slider.dart b/lib/widgets/nt_widgets/single_topic/number_slider.dart index e04b824d..7c619ac7 100644 --- a/lib/widgets/nt_widgets/single_topic/number_slider.dart +++ b/lib/widgets/nt_widgets/single_topic/number_slider.dart @@ -5,7 +5,6 @@ import 'package:dot_cast/dot_cast.dart'; import 'package:provider/provider.dart'; import 'package:syncfusion_flutter_gauges/gauges.dart'; -import 'package:elastic_dashboard/services/nt_connection.dart'; import 'package:elastic_dashboard/services/text_formatter_builder.dart'; import 'package:elastic_dashboard/widgets/dialog_widgets/dialog_text_input.dart'; import 'package:elastic_dashboard/widgets/dialog_widgets/dialog_toggle_switch.dart'; @@ -58,6 +57,8 @@ class NumberSliderModel extends NTWidgetModel { set dragging(value) => _dragging = value; NumberSliderModel({ + required super.ntConnection, + required super.preferences, required super.topic, double minValue = -1.0, double maxValue = 1.0, @@ -71,8 +72,11 @@ class NumberSliderModel extends NTWidgetModel { _maxValue = maxValue, super(); - NumberSliderModel.fromJson({required Map jsonData}) - : super.fromJson(jsonData: jsonData) { + NumberSliderModel.fromJson({ + required super.ntConnection, + required super.preferences, + required Map jsonData, + }) : super.fromJson(jsonData: jsonData) { _minValue = tryCast(jsonData['min_value']) ?? tryCast(jsonData['min']) ?? -1.0; _maxValue = @@ -81,7 +85,9 @@ class NumberSliderModel extends NTWidgetModel { tryCast(jsonData['numOfTickMarks']) ?? 5; - _updateContinuously = tryCast(jsonData['update_continuously']) ?? false; + _updateContinuously = tryCast(jsonData['update_continuously']) ?? + tryCast(jsonData['publish_all']) ?? + false; } @override @@ -91,7 +97,7 @@ class NumberSliderModel extends NTWidgetModel { 'min_value': _minValue, 'max_value': _maxValue, 'divisions': _divisions, - 'publish_all': _updateContinuously, + 'update_continuously': _updateContinuously, }; } @@ -180,7 +186,7 @@ class NumberSliderModel extends NTWidgetModel { } if (publishTopic) { - ntConnection.nt4Client.publishTopic(ntTopic!); + ntConnection.publishTopic(ntTopic!); } ntConnection.updateDataFromTopic(ntTopic!, value); @@ -198,7 +204,7 @@ class NumberSlider extends NTWidget { return StreamBuilder( stream: model.subscription?.periodicStream(), - initialData: ntConnection.getLastAnnouncedValue(model.topic), + initialData: model.ntConnection.getLastAnnouncedValue(model.topic), builder: (context, snapshot) { double value = tryCast(snapshot.data) ?? 0.0; diff --git a/lib/widgets/nt_widgets/single_topic/radial_gauge.dart b/lib/widgets/nt_widgets/single_topic/radial_gauge.dart index ae3fe930..6fe2bebd 100644 --- a/lib/widgets/nt_widgets/single_topic/radial_gauge.dart +++ b/lib/widgets/nt_widgets/single_topic/radial_gauge.dart @@ -5,7 +5,6 @@ import 'package:dot_cast/dot_cast.dart'; import 'package:provider/provider.dart'; import 'package:syncfusion_flutter_gauges/gauges.dart'; -import 'package:elastic_dashboard/services/nt_connection.dart'; import 'package:elastic_dashboard/services/text_formatter_builder.dart'; import 'package:elastic_dashboard/widgets/dialog_widgets/dialog_text_input.dart'; import 'package:elastic_dashboard/widgets/dialog_widgets/dialog_toggle_switch.dart'; @@ -83,6 +82,8 @@ class RadialGaugeModel extends NTWidgetModel { } RadialGaugeModel({ + required super.ntConnection, + required super.preferences, required super.topic, double startAngle = -140.0, double endAngle = 140.0, @@ -104,8 +105,11 @@ class RadialGaugeModel extends NTWidgetModel { _endAngle = endAngle, super(); - RadialGaugeModel.fromJson({required Map jsonData}) - : super.fromJson(jsonData: jsonData) { + RadialGaugeModel.fromJson({ + required super.ntConnection, + required super.preferences, + required Map jsonData, + }) : super.fromJson(jsonData: jsonData) { _startAngle = tryCast(jsonData['start_angle']) ?? _startAngle; _endAngle = tryCast(jsonData['end_angle']) ?? _endAngle; _minValue = tryCast(jsonData['min_value']) ?? _minValue; @@ -298,7 +302,7 @@ class RadialGauge extends NTWidget { return StreamBuilder( stream: model.subscription?.periodicStream(yieldAll: false), - initialData: ntConnection.getLastAnnouncedValue(model.topic), + initialData: model.ntConnection.getLastAnnouncedValue(model.topic), builder: (context, snapshot) { double value = tryCast(snapshot.data) ?? 0.0; diff --git a/lib/widgets/nt_widgets/single_topic/single_color_view.dart b/lib/widgets/nt_widgets/single_topic/single_color_view.dart index 97e44914..ba1cc780 100644 --- a/lib/widgets/nt_widgets/single_topic/single_color_view.dart +++ b/lib/widgets/nt_widgets/single_topic/single_color_view.dart @@ -3,7 +3,6 @@ import 'package:flutter/material.dart'; import 'package:dot_cast/dot_cast.dart'; import 'package:provider/provider.dart'; -import 'package:elastic_dashboard/services/nt_connection.dart'; import 'package:elastic_dashboard/widgets/nt_widgets/nt_widget.dart'; class SingleColorView extends NTWidget { @@ -17,7 +16,7 @@ class SingleColorView extends NTWidget { return StreamBuilder( stream: model.subscription?.periodicStream(yieldAll: false), - initialData: ntConnection.getLastAnnouncedValue(model.topic), + initialData: model.ntConnection.getLastAnnouncedValue(model.topic), builder: (context, snapshot) { String hexString = tryCast(snapshot.data) ?? ''; diff --git a/lib/widgets/nt_widgets/single_topic/text_display.dart b/lib/widgets/nt_widgets/single_topic/text_display.dart index 90158a26..b71ff55e 100644 --- a/lib/widgets/nt_widgets/single_topic/text_display.dart +++ b/lib/widgets/nt_widgets/single_topic/text_display.dart @@ -7,7 +7,6 @@ import 'package:dot_cast/dot_cast.dart'; import 'package:provider/provider.dart'; import 'package:elastic_dashboard/services/nt4_client.dart'; -import 'package:elastic_dashboard/services/nt_connection.dart'; import 'package:elastic_dashboard/widgets/dialog_widgets/dialog_toggle_switch.dart'; import 'package:elastic_dashboard/widgets/nt_widgets/nt_widget.dart'; @@ -29,6 +28,8 @@ class TextDisplayModel extends NTWidgetModel { } TextDisplayModel({ + required super.ntConnection, + required super.preferences, required super.topic, bool showSubmitButton = false, super.dataType, @@ -36,8 +37,11 @@ class TextDisplayModel extends NTWidgetModel { }) : _showSubmitButton = showSubmitButton, super(); - TextDisplayModel.fromJson({required Map jsonData}) - : super.fromJson(jsonData: jsonData) { + TextDisplayModel.fromJson({ + required super.ntConnection, + required super.preferences, + required Map jsonData, + }) : super.fromJson(jsonData: jsonData) { _showSubmitButton = tryCast(jsonData['show_submit_button']) ?? _showSubmitButton; } @@ -59,7 +63,7 @@ class TextDisplayModel extends NTWidgetModel { Map toJson() { return { ...super.toJson(), - 'show_submit_button': _showSubmitButton, + 'show_submit_button': showSubmitButton, }; } @@ -116,7 +120,7 @@ class TextDisplayModel extends NTWidgetModel { } if (publishTopic) { - ntConnection.nt4Client.publishTopic(ntTopic!); + ntConnection.publishTopic(ntTopic!); } if (formattedData != null) { @@ -138,12 +142,11 @@ class TextDisplay extends NTWidget { return StreamBuilder( stream: model.subscription?.periodicStream(), - initialData: ntConnection.getLastAnnouncedValue(model.topic), + initialData: model.ntConnection.getLastAnnouncedValue(model.topic), builder: (context, snapshot) { Object? data = snapshot.data; - if (data?.toString() != model.previousValue?.toString() && - data != null) { + if (data?.toString() != model.previousValue?.toString()) { // Needed to prevent errors Future(() async { String displayString = data.toString(); diff --git a/lib/widgets/nt_widgets/single_topic/toggle_button.dart b/lib/widgets/nt_widgets/single_topic/toggle_button.dart index daee44e2..5ec5c3e4 100644 --- a/lib/widgets/nt_widgets/single_topic/toggle_button.dart +++ b/lib/widgets/nt_widgets/single_topic/toggle_button.dart @@ -3,7 +3,6 @@ import 'package:flutter/material.dart'; import 'package:dot_cast/dot_cast.dart'; import 'package:provider/provider.dart'; -import 'package:elastic_dashboard/services/nt_connection.dart'; import 'package:elastic_dashboard/widgets/nt_widgets/nt_widget.dart'; class ToggleButton extends NTWidget { @@ -17,7 +16,7 @@ class ToggleButton extends NTWidget { return StreamBuilder( stream: model.subscription?.periodicStream(yieldAll: false), - initialData: ntConnection.getLastAnnouncedValue(model.topic), + initialData: model.ntConnection.getLastAnnouncedValue(model.topic), builder: (context, snapshot) { bool value = tryCast(snapshot.data) ?? false; @@ -31,7 +30,7 @@ class ToggleButton extends NTWidget { return GestureDetector( onTapUp: (_) { bool publishTopic = model.ntTopic == null || - !ntConnection.isTopicPublished(model.ntTopic); + !model.ntConnection.isTopicPublished(model.ntTopic); model.createTopicIfNull(); @@ -40,10 +39,10 @@ class ToggleButton extends NTWidget { } if (publishTopic) { - ntConnection.nt4Client.publishTopic(model.ntTopic!); + model.ntConnection.publishTopic(model.ntTopic!); } - ntConnection.updateDataFromTopic(model.ntTopic!, !value); + model.ntConnection.updateDataFromTopic(model.ntTopic!, !value); }, child: Padding( padding: EdgeInsets.symmetric( diff --git a/lib/widgets/nt_widgets/single_topic/toggle_switch.dart b/lib/widgets/nt_widgets/single_topic/toggle_switch.dart index caf5e6d7..0b38758d 100644 --- a/lib/widgets/nt_widgets/single_topic/toggle_switch.dart +++ b/lib/widgets/nt_widgets/single_topic/toggle_switch.dart @@ -3,7 +3,6 @@ import 'package:flutter/material.dart'; import 'package:dot_cast/dot_cast.dart'; import 'package:provider/provider.dart'; -import 'package:elastic_dashboard/services/nt_connection.dart'; import 'package:elastic_dashboard/widgets/nt_widgets/nt_widget.dart'; class ToggleSwitch extends NTWidget { @@ -17,7 +16,7 @@ class ToggleSwitch extends NTWidget { return StreamBuilder( stream: model.subscription?.periodicStream(yieldAll: false), - initialData: ntConnection.getLastAnnouncedValue(model.topic), + initialData: model.ntConnection.getLastAnnouncedValue(model.topic), builder: (context, snapshot) { bool value = tryCast(snapshot.data) ?? false; @@ -25,7 +24,7 @@ class ToggleSwitch extends NTWidget { value: value, onChanged: (bool value) { bool publishTopic = model.ntTopic == null || - !ntConnection.isTopicPublished(model.ntTopic); + !model.ntConnection.isTopicPublished(model.ntTopic); model.createTopicIfNull(); @@ -34,10 +33,10 @@ class ToggleSwitch extends NTWidget { } if (publishTopic) { - ntConnection.nt4Client.publishTopic(model.ntTopic!); + model.ntConnection.publishTopic(model.ntTopic!); } - ntConnection.updateDataFromTopic(model.ntTopic!, value); + model.ntConnection.updateDataFromTopic(model.ntTopic!, value); }, ); }, diff --git a/lib/widgets/nt_widgets/single_topic/voltage_view.dart b/lib/widgets/nt_widgets/single_topic/voltage_view.dart index 5fd6a6bf..c300e63b 100644 --- a/lib/widgets/nt_widgets/single_topic/voltage_view.dart +++ b/lib/widgets/nt_widgets/single_topic/voltage_view.dart @@ -5,7 +5,6 @@ import 'package:dot_cast/dot_cast.dart'; import 'package:provider/provider.dart'; import 'package:syncfusion_flutter_gauges/gauges.dart'; -import 'package:elastic_dashboard/services/nt_connection.dart'; import 'package:elastic_dashboard/services/text_formatter_builder.dart'; import 'package:elastic_dashboard/widgets/dialog_widgets/dialog_dropdown_chooser.dart'; import 'package:elastic_dashboard/widgets/dialog_widgets/dialog_text_input.dart'; @@ -58,6 +57,8 @@ class VoltageViewModel extends NTWidgetModel { } VoltageViewModel({ + required super.ntConnection, + required super.preferences, required super.topic, double minValue = 4.0, double maxValue = 13.0, @@ -73,8 +74,11 @@ class VoltageViewModel extends NTWidgetModel { _minValue = minValue, super(); - VoltageViewModel.fromJson({required Map jsonData}) - : super.fromJson(jsonData: jsonData) { + VoltageViewModel.fromJson({ + required super.ntConnection, + required super.preferences, + required Map jsonData, + }) : super.fromJson(jsonData: jsonData) { _minValue = tryCast(jsonData['min_value']) ?? 4.0; _maxValue = tryCast(jsonData['max_value']) ?? 13.0; _divisions = tryCast(jsonData['divisions']); @@ -86,11 +90,11 @@ class VoltageViewModel extends NTWidgetModel { Map toJson() { return { ...super.toJson(), - 'min_value': _minValue, - 'max_value': _maxValue, - 'divisions': _divisions, - 'inverted': _inverted, - 'orientation': _orientation, + 'min_value': minValue, + 'max_value': maxValue, + if (divisions != null) 'divisions': divisions, + 'inverted': inverted, + 'orientation': orientation, }; } @@ -203,7 +207,7 @@ class VoltageView extends NTWidget { return StreamBuilder( stream: model.subscription?.periodicStream(yieldAll: false), - initialData: ntConnection.getLastAnnouncedValue(model.topic), + initialData: model.ntConnection.getLastAnnouncedValue(model.topic), builder: (context, snapshot) { double voltage = tryCast(snapshot.data) ?? 0.0; diff --git a/lib/widgets/settings_dialog.dart b/lib/widgets/settings_dialog.dart index 68d4d959..9a6d4742 100644 --- a/lib/widgets/settings_dialog.dart +++ b/lib/widgets/settings_dialog.dart @@ -1,7 +1,9 @@ import 'package:flutter/material.dart'; import 'package:flutter/services.dart'; +import 'package:collection/collection.dart'; import 'package:dot_cast/dot_cast.dart'; +import 'package:flex_seed_scheme/flex_seed_scheme.dart'; import 'package:shared_preferences/shared_preferences.dart'; import 'package:elastic_dashboard/services/ip_address_util.dart'; @@ -14,6 +16,15 @@ import 'package:elastic_dashboard/widgets/dialog_widgets/dialog_text_input.dart' import 'package:elastic_dashboard/widgets/dialog_widgets/dialog_toggle_switch.dart'; class SettingsDialog extends StatefulWidget { + final NTConnection ntConnection; + + static final List themeVariants = FlexSchemeVariant.values + .whereNot((variant) => variant == Defaults.themeVariant) + .map((variant) => variant.variantName) + .toList() + ..add(Defaults.defaultVariantName) + ..sort(); + final SharedPreferences preferences; final Function(String? data)? onIPAddressChanged; @@ -28,9 +39,11 @@ class SettingsDialog extends StatefulWidget { final Function(bool value)? onLayoutLock; final Function(String? value)? onDefaultPeriodChanged; final Function(String? value)? onDefaultGraphPeriodChanged; + final Function(FlexSchemeVariant variant)? onThemeVariantChanged; const SettingsDialog({ super.key, + required this.ntConnection, required this.preferences, this.onTeamNumberChanged, this.onIPAddressModeChanged, @@ -44,6 +57,7 @@ class SettingsDialog extends StatefulWidget { this.onLayoutLock, this.onDefaultPeriodChanged, this.onDefaultGraphPeriodChanged, + this.onThemeVariantChanged, }); @override @@ -58,7 +72,7 @@ class _SettingsDialogState extends State { shape: RoundedRectangleBorder(borderRadius: BorderRadius.circular(15.0)), content: Container( constraints: const BoxConstraints( - maxHeight: 275, + maxHeight: 350, maxWidth: 725, ), child: Row( @@ -101,29 +115,71 @@ class _SettingsDialogState extends State { List _generalSettings() { Color currentColor = Color(widget.preferences.getInt(PrefKeys.teamColor) ?? Colors.blueAccent.value); + + // Safety feature to prevent theme variants dropdown from not rendering if the current selection doesn't exist + List? themeVariantsOverride; + if (!SettingsDialog.themeVariants + .contains(widget.preferences.getString(PrefKeys.themeVariant)) && + widget.preferences.getString(PrefKeys.themeVariant) != null) { + // Weird way of copying the list + themeVariantsOverride = SettingsDialog.themeVariants.toList() + ..add(widget.preferences.getString(PrefKeys.themeVariant)!) + ..sort(); + themeVariantsOverride = Set.of(themeVariantsOverride).toList(); + } + return [ - Row( + Column( mainAxisAlignment: MainAxisAlignment.spaceEvenly, children: [ - Flexible( - child: DialogTextInput( - initialText: - widget.preferences.getInt(PrefKeys.teamNumber)?.toString() ?? - Settings.teamNumber.toString(), - label: 'Team Number', - onSubmit: (data) async { - await widget.onTeamNumberChanged?.call(data); - setState(() {}); - }, - formatter: FilteringTextInputFormatter.digitsOnly, - ), + Row( + children: [ + Expanded( + child: DialogTextInput( + initialText: widget.preferences + .getInt(PrefKeys.teamNumber) + ?.toString() ?? + Defaults.teamNumber.toString(), + label: 'Team Number', + onSubmit: (data) async { + await widget.onTeamNumberChanged?.call(data); + setState(() {}); + }, + formatter: FilteringTextInputFormatter.digitsOnly, + ), + ), + Expanded( + child: DialogColorPicker( + onColorPicked: (color) => widget.onColorChanged?.call(color), + label: 'Team Color', + initialColor: currentColor, + ), + ), + ], ), - Flexible( - child: DialogColorPicker( - onColorPicked: (color) => widget.onColorChanged?.call(color), - label: 'Team Color', - initialColor: currentColor, - ), + Row( + children: [ + const Text('Theme Variant'), + const SizedBox(width: 5), + Flexible( + child: DialogDropdownChooser( + onSelectionChanged: (variantName) { + if (variantName == null) return; + FlexSchemeVariant variant = FlexSchemeVariant.values + .firstWhereOrNull( + (e) => e.variantName == variantName) ?? + FlexSchemeVariant.material3Legacy; + + widget.onThemeVariantChanged?.call(variant); + setState(() {}); + }, + choices: + themeVariantsOverride ?? SettingsDialog.themeVariants, + initialValue: + widget.preferences.getString(PrefKeys.themeVariant) ?? + Defaults.defaultVariantName), + ), + ], ), ], ), @@ -149,21 +205,24 @@ class _SettingsDialogState extends State { setState(() {}); }, choices: IPAddressMode.values, - initialValue: Settings.ipAddressMode, + initialValue: IPAddressMode.fromIndex( + widget.preferences.getInt(PrefKeys.ipAddressMode)), ), const SizedBox(height: 5), StreamBuilder( - stream: ntConnection.dsConnectionStatus(), - initialData: ntConnection.isDSConnected, + stream: widget.ntConnection.dsConnectionStatus(), + initialData: widget.ntConnection.isDSConnected, builder: (context, snapshot) { bool dsConnected = tryCast(snapshot.data) ?? false; return DialogTextInput( - enabled: Settings.ipAddressMode == IPAddressMode.custom || - (Settings.ipAddressMode == IPAddressMode.driverStation && + enabled: widget.preferences.getInt(PrefKeys.ipAddressMode) == + IPAddressMode.custom.index || + (widget.preferences.getInt(PrefKeys.ipAddressMode) == + IPAddressMode.driverStation.index && !dsConnected), initialText: widget.preferences.getString(PrefKeys.ipAddress) ?? - Settings.ipAddress, + Defaults.ipAddress, label: 'IP Address', onSubmit: (String? data) async { await widget.onIPAddressChanged?.call(data); @@ -187,7 +246,7 @@ class _SettingsDialogState extends State { Flexible( child: DialogToggleSwitch( initialValue: widget.preferences.getBool(PrefKeys.showGrid) ?? - Settings.showGrid, + Defaults.showGrid, label: 'Show Grid', onToggle: (value) { setState(() { @@ -200,7 +259,7 @@ class _SettingsDialogState extends State { child: DialogTextInput( initialText: widget.preferences.getInt(PrefKeys.gridSize)?.toString() ?? - Settings.gridSize.toString(), + Defaults.gridSize.toString(), label: 'Grid Size', onSubmit: (value) async { await widget.onGridSizeChanged?.call(value); @@ -218,10 +277,10 @@ class _SettingsDialogState extends State { Flexible( flex: 2, child: DialogTextInput( - initialText: widget.preferences - .getDouble(PrefKeys.cornerRadius) - ?.toString() ?? - Settings.cornerRadius.toString(), + initialText: + (widget.preferences.getDouble(PrefKeys.cornerRadius) ?? + Defaults.cornerRadius.toString()) + .toString(), label: 'Corner Radius', onSubmit: (value) { setState(() { @@ -236,7 +295,7 @@ class _SettingsDialogState extends State { child: DialogToggleSwitch( initialValue: widget.preferences.getBool(PrefKeys.autoResizeToDS) ?? - Settings.autoResizeToDS, + Defaults.autoResizeToDS, label: 'Resize to Driver Station Height', onToggle: (value) { setState(() { @@ -269,7 +328,7 @@ class _SettingsDialogState extends State { flex: 4, child: DialogToggleSwitch( initialValue: widget.preferences.getBool(PrefKeys.layoutLocked) ?? - Settings.layoutLocked, + Defaults.layoutLocked, label: 'Lock Layout', onToggle: (value) { setState(() { @@ -298,7 +357,7 @@ class _SettingsDialogState extends State { child: DialogTextInput( initialText: (widget.preferences.getDouble(PrefKeys.defaultPeriod) ?? - Settings.defaultPeriod) + Defaults.defaultPeriod) .toString(), label: 'Default Period', onSubmit: (value) async { @@ -312,7 +371,7 @@ class _SettingsDialogState extends State { child: DialogTextInput( initialText: (widget.preferences .getDouble(PrefKeys.defaultGraphPeriod) ?? - Settings.defaultGraphPeriod) + Defaults.defaultGraphPeriod) .toString(), label: 'Default Graph Period', onSubmit: (value) async { diff --git a/lib/widgets/tab_grid.dart b/lib/widgets/tab_grid.dart index 91d50236..5ebb7be2 100644 --- a/lib/widgets/tab_grid.dart +++ b/lib/widgets/tab_grid.dart @@ -6,6 +6,7 @@ import 'package:dot_cast/dot_cast.dart'; import 'package:flutter_box_transform/flutter_box_transform.dart'; import 'package:flutter_context_menu/flutter_context_menu.dart'; import 'package:provider/provider.dart'; +import 'package:shared_preferences/shared_preferences.dart'; import 'package:elastic_dashboard/services/nt_connection.dart'; import 'package:elastic_dashboard/services/settings.dart'; @@ -21,25 +22,25 @@ import 'draggable_containers/models/widget_container_model.dart'; // Used to refresh the tab grid when a widget is added or removed // This doesn't use a stateful widget since everything has to be rendered at program startup or data will be lost class TabGridModel extends ChangeNotifier { - void notify() { - notifyListeners(); - } -} - -class TabGrid extends StatelessWidget { - static Map? _copyJsonData; + final NTConnection ntConnection; + final SharedPreferences preferences; final List _widgetModels = []; + static Map? copyJsonData; + MapEntry? _containerDraggingIn; + BuildContext? tabGridContext; final VoidCallback onAddWidgetPressed; - TabGridModel? model; - - TabGrid({super.key, required this.onAddWidgetPressed}); + TabGridModel( + {required this.ntConnection, + required this.preferences, + required this.onAddWidgetPressed}); - TabGrid.fromJson({ - super.key, + TabGridModel.fromJson({ + required this.ntConnection, + required this.preferences, required Map jsonData, required this.onAddWidgetPressed, Function(String message)? onJsonLoadingWarning, @@ -59,6 +60,8 @@ class TabGrid extends StatelessWidget { for (Map containerData in jsonData['containers']) { _widgetModels.add( NTWidgetContainerModel.fromJson( + ntConnection: ntConnection, + preferences: preferences, enabled: ntConnection.isNT4Connected, jsonData: containerData, onJsonLoadingWarning: onJsonLoadingWarning, @@ -81,7 +84,16 @@ class TabGrid extends StatelessWidget { switch (layoutData['type']) { case 'List Layout': widget = ListLayoutModel.fromJson( + preferences: preferences, jsonData: layoutData, + ntWidgetBuilder: (preferences, jsonData, enabled, + {onJsonLoadingWarning}) => + NTWidgetContainerModel.fromJson( + ntConnection: ntConnection, + jsonData: jsonData, + preferences: preferences, + onJsonLoadingWarning: onJsonLoadingWarning, + ), enabled: ntConnection.isNT4Connected, tabGrid: this, onDragCancel: _layoutContainerOnDragCancel, @@ -115,13 +127,12 @@ class TabGrid extends StatelessWidget { } Offset getLocalPosition(Offset globalPosition) { - BuildContext? context = (key as GlobalKey).currentContext; - - if (context == null) { + if (tabGridContext == null) { return Offset.zero; } - RenderBox? ancestor = context.findAncestorRenderObjectOfType(); + RenderBox? ancestor = + tabGridContext!.findAncestorRenderObjectOfType(); Offset localPosition = ancestor!.globalToLocal(globalPosition); @@ -144,10 +155,9 @@ class TabGrid extends StatelessWidget { /// /// This only applies to widgets that already have a place on the grid bool isValidMoveLocation(WidgetContainerModel widget, Rect location) { - BuildContext? context = (key as GlobalKey).currentContext; Size? gridSize; - if (context != null) { - gridSize = MediaQuery.of(context).size; + if (tabGridContext != null) { + gridSize = MediaQuery.of(tabGridContext!).size; } for (WidgetContainerModel container in _widgetModels) { @@ -303,13 +313,13 @@ class TabGrid extends StatelessWidget { if (previewWidth < model.minWidth) { previewWidth = DraggableWidgetContainer.snapToGrid( constrainedRect.width.clamp(model.minWidth, double.infinity) + - Settings.gridSize); + (preferences.getInt(PrefKeys.gridSize) ?? Defaults.gridSize)); } if (previewHeight < model.minHeight) { previewHeight = DraggableWidgetContainer.snapToGrid( constrainedRect.height.clamp(model.minHeight, double.infinity) + - Settings.gridSize); + (preferences.getInt(PrefKeys.gridSize) ?? Defaults.gridSize)); } Rect preview = @@ -524,12 +534,17 @@ class TabGrid extends StatelessWidget { ListLayoutModel createListLayout( {String title = 'List Layout', List? children}) { return ListLayoutModel( + preferences: preferences, title: title, initialPosition: Rect.fromLTWH( 0.0, 0.0, - Settings.gridSize.toDouble() * 2, - Settings.gridSize.toDouble() * 2, + (preferences.getInt(PrefKeys.gridSize) ?? Defaults.gridSize) + .toDouble() * + 2, + (preferences.getInt(PrefKeys.gridSize) ?? Defaults.gridSize) + .toDouble() * + 2, ), children: children, minWidth: 128.0, @@ -586,7 +601,16 @@ class TabGrid extends StatelessWidget { case 'List Layout': _widgetModels.add( ListLayoutModel.fromJson( + preferences: preferences, jsonData: widgetData, + ntWidgetBuilder: (preferences, jsonData, enabled, + {onJsonLoadingWarning}) => + NTWidgetContainerModel.fromJson( + ntConnection: ntConnection, + jsonData: jsonData, + preferences: preferences, + onJsonLoadingWarning: onJsonLoadingWarning, + ), enabled: ntConnection.isNT4Connected, tabGrid: this, onDragCancel: _layoutContainerOnDragCancel, @@ -599,6 +623,8 @@ class TabGrid extends StatelessWidget { } else { _widgetModels.add( NTWidgetContainerModel.fromJson( + ntConnection: ntConnection, + preferences: preferences, enabled: ntConnection.isNT4Connected, jsonData: widgetData, ), @@ -650,7 +676,7 @@ class TabGrid extends StatelessWidget { } void copyWidget(WidgetContainerModel widget) { - _copyJsonData = widget.toJson(); + copyJsonData = widget.toJson(); } void lockLayout() { @@ -674,12 +700,6 @@ class TabGrid extends StatelessWidget { _widgetModels.clear(); } - void refresh() { - Future(() async { - model?.notify(); - }); - } - void resizeGrid(int oldSize, int newSize) { for (WidgetContainerModel widget in _widgetModels) { widget.updateGridSize(oldSize, newSize); @@ -695,37 +715,51 @@ class TabGrid extends StatelessWidget { }); } + void refresh() { + notifyListeners(); + } +} + +class TabGrid extends StatelessWidget { + const TabGrid({super.key}); + @override Widget build(BuildContext context) { - model = context.watch(); + TabGridModel model = context.watch(); - Widget getWidgetFromModel(WidgetContainerModel model) { - if (model is NTWidgetContainerModel) { + model.tabGridContext = context; + + Widget getWidgetFromModel(WidgetContainerModel widgetModel) { + if (widgetModel is NTWidgetContainerModel) { return ChangeNotifierProvider.value( - value: model, + value: widgetModel, child: DraggableNTWidgetContainer( - key: model.key, - tabGrid: this, - onUpdate: _ntContainerOnUpdate, - onDragBegin: _ntContainerOnDragBegin, - onDragEnd: _ntContainerOnDragEnd, - onDragCancel: _ntContainerOnDragCancel, - onResizeBegin: _ntContainerOnResizeBegin, - onResizeEnd: _ntContainerOnResizeEnd, + key: widgetModel.key, + updateFunctions: ( + onUpdate: model._ntContainerOnUpdate, + onDragBegin: model._ntContainerOnDragBegin, + onDragEnd: model._ntContainerOnDragEnd, + onDragCancel: model._ntContainerOnDragCancel, + onResizeBegin: model._ntContainerOnResizeBegin, + onResizeEnd: model._ntContainerOnResizeEnd, + isValidMoveLocation: model.isValidMoveLocation, + ), ), ); - } else if (model is ListLayoutModel) { + } else if (widgetModel is ListLayoutModel) { return ChangeNotifierProvider.value( - value: model, + value: widgetModel, child: DraggableListLayout( - key: model.key, - tabGrid: this, - onUpdate: _layoutContainerOnUpdate, - onDragBegin: _layoutContainerOnDragBegin, - onDragEnd: _layoutContainerOnDragEnd, - onDragCancel: _layoutContainerOnDragCancel, - onResizeBegin: _layoutContainerOnResizeBegin, - onResizeEnd: _layoutContainerOnResizeEnd, + key: widgetModel.key, + updateFunctions: ( + onUpdate: model._layoutContainerOnUpdate, + onDragBegin: model._layoutContainerOnDragBegin, + onDragEnd: model._layoutContainerOnDragEnd, + onDragCancel: model._layoutContainerOnDragCancel, + onResizeBegin: model._layoutContainerOnResizeBegin, + onResizeEnd: model._layoutContainerOnResizeEnd, + isValidMoveLocation: model.isValidMoveLocation, + ), ), ); } @@ -737,7 +771,7 @@ class TabGrid extends StatelessWidget { List draggingInWidgets = []; List previewOutlines = []; - for (WidgetContainerModel container in _widgetModels) { + for (WidgetContainerModel container in model._widgetModels) { if (container.dragging) { draggingWidgets.add( Positioned( @@ -755,7 +789,7 @@ class TabGrid extends StatelessWidget { ); } else { LayoutContainerModel? layoutContainer = - getLayoutAtLocation(container.cursorGlobalLocation); + model.getLayoutAtLocation(container.cursorGlobalLocation); if (layoutContainer == null) { previewOutlines.add( @@ -773,8 +807,9 @@ class TabGrid extends StatelessWidget { child: Container( decoration: BoxDecoration( color: Colors.white.withOpacity(0.25), - borderRadius: - BorderRadius.circular(Settings.cornerRadius), + borderRadius: BorderRadius.circular( + model.preferences.getDouble(PrefKeys.cornerRadius) ?? + Defaults.cornerRadius), border: Border.all(color: Colors.yellow, width: 5.0), ), ), @@ -788,7 +823,8 @@ class TabGrid extends StatelessWidget { dashboardWidgets.add( GestureDetector( onSecondaryTapUp: (details) { - if (Settings.layoutLocked) { + if (model.preferences.getBool(PrefKeys.layoutLocked) ?? + Defaults.layoutLocked) { return; } List menuEntries = [ @@ -809,13 +845,13 @@ class TabGrid extends StatelessWidget { label: 'Copy', icon: Icons.copy_outlined, onSelected: () { - copyWidget(container); + model.copyWidget(container); }), MenuItem( label: 'Remove', icon: Icons.delete_outlined, onSelected: () { - removeWidget(container); + model.removeWidget(container); }), ]; @@ -846,8 +882,8 @@ class TabGrid extends StatelessWidget { } // Also render any containers that are being dragged into the grid - if (_containerDraggingIn != null) { - WidgetContainerModel container = _containerDraggingIn!.key; + if (model._containerDraggingIn != null) { + WidgetContainerModel container = model._containerDraggingIn!.key; draggingWidgets.add( Positioned( @@ -867,15 +903,16 @@ class TabGrid extends StatelessWidget { Rect previewLocation = Rect.fromLTWH(previewX, previewY, container.displayRect.width, container.displayRect.height); - bool validLocation = isValidMoveLocation(container, previewLocation) || - isValidLayoutLocation(container.cursorGlobalLocation); + bool validLocation = + model.isValidMoveLocation(container, previewLocation) || + model.isValidLayoutLocation(container.cursorGlobalLocation); Color borderColor = (validLocation) ? Colors.lightGreenAccent.shade400 : Colors.red; - if (isValidLayoutLocation(container.cursorGlobalLocation)) { + if (model.isValidLayoutLocation(container.cursorGlobalLocation)) { LayoutContainerModel layoutContainer = - getLayoutAtLocation(container.cursorGlobalLocation)!; + model.getLayoutAtLocation(container.cursorGlobalLocation)!; previewLocation = layoutContainer.displayRect; @@ -893,7 +930,9 @@ class TabGrid extends StatelessWidget { color: (validLocation) ? Colors.white.withOpacity(0.25) : Colors.black.withOpacity(0.1), - borderRadius: BorderRadius.circular(Settings.cornerRadius), + borderRadius: BorderRadius.circular( + model.preferences.getDouble(PrefKeys.cornerRadius) ?? + Defaults.cornerRadius), border: Border.all(color: borderColor, width: 5.0), ), ), @@ -905,7 +944,8 @@ class TabGrid extends StatelessWidget { behavior: HitTestBehavior.translucent, onTap: () {}, onSecondaryTapUp: (details) { - if (Settings.layoutLocked) { + if (model.preferences.getBool(PrefKeys.layoutLocked) ?? + Defaults.layoutLocked) { return; } @@ -913,22 +953,23 @@ class TabGrid extends StatelessWidget { MenuItem( label: 'Add Widget', icon: Icons.add, - onSelected: () => onAddWidgetPressed.call(), + onSelected: () => model.onAddWidgetPressed.call(), ), MenuItem( label: 'Clear Layout', icon: Icons.clear, - onSelected: () => clearWidgets(context), + onSelected: () => model.clearWidgets(context), ), ]; - if (_copyJsonData != null) { + if (TabGridModel.copyJsonData != null) { contextMenuEntries.add( MenuItem( label: 'Paste', icon: Icons.paste_outlined, onSelected: () { - pasteWidget(_copyJsonData, details.localPosition); + pasteWidget( + model, TabGridModel.copyJsonData, details.localPosition); }, ), ); @@ -965,14 +1006,16 @@ class TabGrid extends StatelessWidget { ); } - void pasteWidget(Map? widgetJson, Offset localPosition) { + void pasteWidget(TabGridModel grid, Map? widgetJson, + Offset localPosition) { if (widgetJson == null) return; + int gridSize = + grid.preferences.getInt(PrefKeys.gridSize) ?? Defaults.gridSize; + // Put the top left corner of the widget in the square the user pastes it in - double snappedX = - (localPosition.dx ~/ Settings.gridSize) * Settings.gridSize.toDouble(); - double snappedY = - (localPosition.dy ~/ Settings.gridSize) * Settings.gridSize.toDouble(); + double snappedX = (localPosition.dx ~/ gridSize) * gridSize.toDouble(); + double snappedY = (localPosition.dy ~/ gridSize) * gridSize.toDouble(); widgetJson['x'] = snappedX; widgetJson['y'] = snappedY; @@ -984,24 +1027,37 @@ class TabGrid extends StatelessWidget { widgetJson['height'], ); - if (isValidLocation(pasteLocation)) { - WidgetContainerModel copiedWidget = createWidgetFromJson(widgetJson); + if (grid.isValidLocation(pasteLocation)) { + WidgetContainerModel copiedWidget = + createWidgetFromJson(grid, widgetJson); - _widgetModels.add(copiedWidget); - refresh(); + grid._widgetModels.add(copiedWidget); + grid.refresh(); } } - WidgetContainerModel createWidgetFromJson(Map json) { + WidgetContainerModel createWidgetFromJson( + TabGridModel grid, Map json) { if (json['type'] == 'List Layout') { return ListLayoutModel.fromJson( + preferences: grid.preferences, jsonData: json, - tabGrid: this, - onDragCancel: _layoutContainerOnDragCancel, + tabGrid: grid, + ntWidgetBuilder: (preferences, jsonData, enabled, + {onJsonLoadingWarning}) => + NTWidgetContainerModel.fromJson( + ntConnection: grid.ntConnection, + jsonData: jsonData, + preferences: preferences, + onJsonLoadingWarning: onJsonLoadingWarning, + ), + onDragCancel: grid._layoutContainerOnDragCancel, ); } else { return NTWidgetContainerModel.fromJson( - enabled: ntConnection.isNT4Connected, + ntConnection: grid.ntConnection, + preferences: grid.preferences, + enabled: grid.ntConnection.isNT4Connected, jsonData: json, ); } diff --git a/pubspec.yaml b/pubspec.yaml index 65b96118..4f9e7ee0 100644 --- a/pubspec.yaml +++ b/pubspec.yaml @@ -67,6 +67,8 @@ flutter: import_sorter: comments: false + ignored_files: + - /*.mocks.dart flutter_launcher_icons: image_path: "assets/logos/logo.png" diff --git a/test/main_test.dart b/test/main_test.dart index c3e4cca7..36b29e0f 100644 --- a/test/main_test.dart +++ b/test/main_test.dart @@ -13,8 +13,6 @@ import 'test_util.dart'; void main() { TestWidgetsFlutterBinding.ensureInitialized(); - setupMockOfflineNT4(); - testWidgets('Full app test', (widgetTester) async { FlutterError.onError = ignoreOverflowErrors; @@ -33,8 +31,9 @@ void main() { await widgetTester.pumpWidget( Elastic( - version: '0.0.0.0', + ntConnection: createMockOfflineNT4(), preferences: preferences, + version: '0.0.0.0', ), ); diff --git a/test/pages/dashboard_page_test.dart b/test/pages/dashboard_page_test.dart index 77e6da29..a4e1bc38 100644 --- a/test/pages/dashboard_page_test.dart +++ b/test/pages/dashboard_page_test.dart @@ -3,7 +3,9 @@ import 'dart:io'; import 'package:flutter/gestures.dart'; import 'package:flutter/material.dart'; +import 'package:flutter/services.dart'; +import 'package:elegant_notification/elegant_notification.dart'; import 'package:flutter_test/flutter_test.dart'; import 'package:mockito/mockito.dart'; import 'package:shared_preferences/shared_preferences.dart'; @@ -11,8 +13,8 @@ import 'package:titlebar_buttons/titlebar_buttons.dart'; import 'package:elastic_dashboard/pages/dashboard_page.dart'; import 'package:elastic_dashboard/services/field_images.dart'; +import 'package:elastic_dashboard/services/hotkey_manager.dart'; import 'package:elastic_dashboard/services/nt4_client.dart'; -import 'package:elastic_dashboard/services/nt_connection.dart'; import 'package:elastic_dashboard/services/settings.dart'; import 'package:elastic_dashboard/widgets/custom_appbar.dart'; import 'package:elastic_dashboard/widgets/dialog_widgets/dialog_text_input.dart'; @@ -53,13 +55,17 @@ void main() { preferences = await SharedPreferences.getInstance(); }); + tearDown(() { + hotKeyManager.tearDown(); + }); + testWidgets('Dashboard page loading offline', (widgetTester) async { FlutterError.onError = ignoreOverflowErrors; - setupMockOfflineNT4(); await widgetTester.pumpWidget( MaterialApp( home: DashboardPage( + ntConnection: createMockOfflineNT4(), preferences: preferences, version: '0.0.0.0', ), @@ -79,11 +85,11 @@ void main() { testWidgets('Dashboard page loading online', (widgetTester) async { FlutterError.onError = ignoreOverflowErrors; - setupMockOnlineNT4(); await widgetTester.pumpWidget( MaterialApp( home: DashboardPage( + ntConnection: createMockOnlineNT4(), preferences: preferences, version: '0.0.0.0', ), @@ -103,11 +109,11 @@ void main() { testWidgets('Save layout (button)', (widgetTester) async { FlutterError.onError = ignoreOverflowErrors; - setupMockOfflineNT4(); await widgetTester.pumpWidget( MaterialApp( home: DashboardPage( + ntConnection: createMockOfflineNT4(), preferences: preferences, version: '0.0.0.0', ), @@ -133,13 +139,38 @@ void main() { expect(jsonString, preferences.getString(PrefKeys.layout)); }); + testWidgets('Save layout (shortcut)', (widgetTester) async { + FlutterError.onError = ignoreOverflowErrors; + + await widgetTester.pumpWidget( + MaterialApp( + home: DashboardPage( + ntConnection: createMockOfflineNT4(), + preferences: preferences, + version: '0.0.0.0', + ), + ), + ); + + await widgetTester.pumpAndSettle(); + + await widgetTester.sendKeyDownEvent(LogicalKeyboardKey.control); + + await widgetTester.sendKeyDownEvent(LogicalKeyboardKey.keyS); + await widgetTester.sendKeyUpEvent(LogicalKeyboardKey.keyS); + await widgetTester.pumpAndSettle(); + + expect(jsonString, preferences.getString(PrefKeys.layout)); + }); + testWidgets('Add widget dialog (widgets)', (widgetTester) async { FlutterError.onError = ignoreOverflowErrors; - setupMockOnlineNT4(); + createMockOnlineNT4(); await widgetTester.pumpWidget( MaterialApp( home: DashboardPage( + ntConnection: createMockOnlineNT4(), preferences: preferences, version: '0.0.0.0', ), @@ -198,11 +229,11 @@ void main() { testWidgets('Add widget dialog (layouts)', (widgetTester) async { FlutterError.onError = ignoreOverflowErrors; - setupMockOnlineNT4(); await widgetTester.pumpWidget( MaterialApp( home: DashboardPage( + ntConnection: createMockOnlineNT4(), preferences: preferences, version: '0.0.0.0', ), @@ -253,11 +284,11 @@ void main() { testWidgets('List Layouts', (widgetTester) async { FlutterError.onError = ignoreOverflowErrors; - setupMockOnlineNT4(); await widgetTester.pumpWidget( MaterialApp( home: DashboardPage( + ntConnection: createMockOnlineNT4(), preferences: preferences, version: '0.0.0.0', ), @@ -348,7 +379,6 @@ void main() { // A custom mock is set up to reproduce behavior when actually running final mockNT4Connection = MockNTConnection(); - final mockNT4Client = MockNT4Client(); final mockSubscription = MockNT4Subscription(); when(mockNT4Connection.isNT4Connected).thenReturn(true); @@ -359,13 +389,13 @@ void main() { when(mockSubscription.periodicStream()) .thenAnswer((_) => Stream.value(null)); - when(mockNT4Client.addTopicAnnounceListener(any)) + when(mockSubscription.listen(any)).thenAnswer((realInvocation) {}); + + when(mockNT4Connection.addTopicAnnounceListener(any)) .thenAnswer((realInvocation) { fakeAnnounceCallbacks.add(realInvocation.positionalArguments[0]); }); - when(mockNT4Connection.nt4Client).thenReturn(mockNT4Client); - when(mockNT4Connection.getLastAnnouncedValue(any)).thenReturn(null); when(mockNT4Connection.subscribe(any, any)).thenReturn(mockSubscription); @@ -400,11 +430,10 @@ void main() { '/Shuffleboard/Test-Tab/Shuffleboard Test Layout/.type')) .thenAnswer((realInvocation) => Future.value('ShuffleboardLayout')); - NTConnection.instance = mockNT4Connection; - await widgetTester.pumpWidget( MaterialApp( home: DashboardPage( + ntConnection: mockNT4Connection, preferences: preferences, version: '0.0.0.0', ), @@ -479,11 +508,11 @@ void main() { testWidgets('About dialog', (widgetTester) async { FlutterError.onError = ignoreOverflowErrors; - setupMockOfflineNT4(); await widgetTester.pumpWidget( MaterialApp( home: DashboardPage( + ntConnection: createMockOfflineNT4(), preferences: preferences, version: '0.0.0.0', ), @@ -511,11 +540,11 @@ void main() { testWidgets('Changing tabs', (widgetTester) async { FlutterError.onError = ignoreOverflowErrors; - setupMockOfflineNT4(); await widgetTester.pumpWidget( MaterialApp( home: DashboardPage( + ntConnection: createMockOfflineNT4(), preferences: preferences, version: '0.0.0.0', ), @@ -540,11 +569,11 @@ void main() { testWidgets('Creating new tab', (widgetTester) async { FlutterError.onError = ignoreOverflowErrors; - setupMockOfflineNT4(); await widgetTester.pumpWidget( MaterialApp( home: DashboardPage( + ntConnection: createMockOfflineNT4(), preferences: preferences, version: '0.0.0.0', ), @@ -568,13 +597,39 @@ void main() { expect(find.byType(TabGrid, skipOffstage: false), findsNWidgets(3)); }); + testWidgets('Creating new tab (shortcut)', (widgetTester) async { + FlutterError.onError = ignoreOverflowErrors; + + await widgetTester.pumpWidget( + MaterialApp( + home: DashboardPage( + ntConnection: createMockOfflineNT4(), + preferences: preferences, + version: '0.0.0.0', + ), + ), + ); + + await widgetTester.pumpAndSettle(); + + expect(find.byType(TabGrid, skipOffstage: false), findsNWidgets(2)); + + await widgetTester.sendKeyDownEvent(LogicalKeyboardKey.control); + await widgetTester.sendKeyDownEvent(LogicalKeyboardKey.keyT); + await widgetTester.sendKeyUpEvent(LogicalKeyboardKey.keyT); + + await widgetTester.pumpAndSettle(); + + expect(find.byType(TabGrid, skipOffstage: false), findsNWidgets(3)); + }); + testWidgets('Closing tab', (widgetTester) async { FlutterError.onError = ignoreOverflowErrors; - setupMockOfflineNT4(); await widgetTester.pumpWidget( MaterialApp( home: DashboardPage( + ntConnection: createMockOfflineNT4(), preferences: preferences, version: '0.0.0.0', ), @@ -610,13 +665,49 @@ void main() { expect(find.byType(TabGrid, skipOffstage: false), findsNWidgets(1)); }); + testWidgets('Closing tab (shortcut)', (widgetTester) async { + FlutterError.onError = ignoreOverflowErrors; + + await widgetTester.pumpWidget( + MaterialApp( + home: DashboardPage( + ntConnection: createMockOfflineNT4(), + preferences: preferences, + version: '0.0.0.0', + ), + ), + ); + + await widgetTester.pumpAndSettle(); + + expect(find.byType(TabGrid, skipOffstage: false), findsNWidgets(2)); + + await widgetTester.sendKeyDownEvent(LogicalKeyboardKey.control); + await widgetTester.sendKeyDownEvent(LogicalKeyboardKey.keyW); + await widgetTester.sendKeyUpEvent(LogicalKeyboardKey.keyW); + + await widgetTester.pumpAndSettle(); + + expect(find.text('Confirm Tab Close', skipOffstage: false), findsOneWidget); + + final confirmButton = + find.widgetWithText(TextButton, 'OK', skipOffstage: false); + + expect(confirmButton, findsOneWidget); + + await widgetTester.tap(confirmButton); + await widgetTester.pumpAndSettle(); + + expect(find.byType(TabGrid, skipOffstage: false), findsNWidgets(1)); + }); + testWidgets('Reordering tabs', (widgetTester) async { FlutterError.onError = ignoreOverflowErrors; - setupMockOfflineNT4(); await widgetTester.pumpWidget( MaterialApp( home: DashboardPage( + ntConnection: createMockOfflineNT4(), preferences: preferences, version: '0.0.0.0', ), @@ -668,13 +759,179 @@ void main() { expect(editableTabBarWidget().currentIndex, 0); }); + testWidgets('Reordering tabs (shortcut)', (widgetTester) async { + FlutterError.onError = ignoreOverflowErrors; + + await widgetTester.pumpWidget( + MaterialApp( + home: DashboardPage( + ntConnection: createMockOfflineNT4(), + preferences: preferences, + version: '0.0.0.0', + ), + ), + ); + + await widgetTester.pumpAndSettle(); + + expect(find.byType(TabGrid, skipOffstage: false), findsNWidgets(2)); + + final editableTabBar = find.byType(EditableTabBar); + + expect(editableTabBar, findsOneWidget); + + editableTabBarWidget() => + (editableTabBar.evaluate().first.widget as EditableTabBar); + + expect(editableTabBarWidget().currentIndex, 0); + + await widgetTester.sendKeyDownEvent(LogicalKeyboardKey.control); + + await widgetTester.sendKeyDownEvent(LogicalKeyboardKey.arrowLeft); + await widgetTester.sendKeyUpEvent(LogicalKeyboardKey.arrowLeft); + await widgetTester.pumpAndSettle(); + + expect(editableTabBarWidget().currentIndex, 0, + reason: 'Tab index should not change since index is 0'); + + await widgetTester.sendKeyDownEvent(LogicalKeyboardKey.arrowRight); + await widgetTester.sendKeyUpEvent(LogicalKeyboardKey.arrowRight); + await widgetTester.pumpAndSettle(); + + expect(editableTabBarWidget().currentIndex, 1); + + await widgetTester.sendKeyDownEvent(LogicalKeyboardKey.arrowRight); + await widgetTester.sendKeyUpEvent(LogicalKeyboardKey.arrowRight); + await widgetTester.pumpAndSettle(); + + expect(editableTabBarWidget().currentIndex, 1, + reason: + 'Tab index should not change since index is equal to number of tabs'); + + await widgetTester.sendKeyDownEvent(LogicalKeyboardKey.arrowLeft); + await widgetTester.sendKeyUpEvent(LogicalKeyboardKey.arrowLeft); + await widgetTester.pumpAndSettle(); + + expect(editableTabBarWidget().currentIndex, 0); + }); + + testWidgets('Navigate tabs left right (shortcut)', (widgetTester) async { + FlutterError.onError = ignoreOverflowErrors; + + await widgetTester.pumpWidget( + MaterialApp( + home: DashboardPage( + ntConnection: createMockOfflineNT4(), + preferences: preferences, + version: '0.0.0.0', + ), + ), + ); + + await widgetTester.pumpAndSettle(); + + expect(find.byType(TabGrid, skipOffstage: false), findsNWidgets(2)); + + final editableTabBar = find.byType(EditableTabBar); + + expect(editableTabBar, findsOneWidget); + + editableTabBarWidget() => + (editableTabBar.evaluate().first.widget as EditableTabBar); + + expect(editableTabBarWidget().currentIndex, 0); + + await widgetTester.sendKeyDownEvent(LogicalKeyboardKey.control); + await widgetTester.sendKeyDownEvent(LogicalKeyboardKey.shift); + + await widgetTester.sendKeyDownEvent(LogicalKeyboardKey.tab); + await widgetTester.sendKeyUpEvent(LogicalKeyboardKey.tab); + await widgetTester.pumpAndSettle(); + + expect(editableTabBarWidget().currentIndex, 1, + reason: 'Tab index should roll over'); + + await widgetTester.sendKeyUpEvent(LogicalKeyboardKey.shift); + + await widgetTester.sendKeyDownEvent(LogicalKeyboardKey.tab); + await widgetTester.sendKeyUpEvent(LogicalKeyboardKey.tab); + await widgetTester.pumpAndSettle(); + + expect(editableTabBarWidget().currentIndex, 0, + reason: 'Tab index should roll back over to 0'); + + await widgetTester.sendKeyDownEvent(LogicalKeyboardKey.tab); + await widgetTester.sendKeyUpEvent(LogicalKeyboardKey.tab); + await widgetTester.pumpAndSettle(); + + expect(editableTabBarWidget().currentIndex, 1, + reason: 'Tab index should increase to 1 (no rollover)'); + + await widgetTester.sendKeyDownEvent(LogicalKeyboardKey.shift); + await widgetTester.sendKeyDownEvent(LogicalKeyboardKey.tab); + await widgetTester.sendKeyUpEvent(LogicalKeyboardKey.tab); + await widgetTester.pumpAndSettle(); + + expect(editableTabBarWidget().currentIndex, 0); + }); + + testWidgets('Navigate to specific tabs', (widgetTester) async { + FlutterError.onError = ignoreOverflowErrors; + + await widgetTester.pumpWidget( + MaterialApp( + home: DashboardPage( + ntConnection: createMockOfflineNT4(), + preferences: preferences, + version: '0.0.0.0', + ), + ), + ); + + await widgetTester.pumpAndSettle(); + + expect(find.byType(TabGrid, skipOffstage: false), findsNWidgets(2)); + + final editableTabBar = find.byType(EditableTabBar); + + expect(editableTabBar, findsOneWidget); + + editableTabBarWidget() => + (editableTabBar.evaluate().first.widget as EditableTabBar); + + expect(editableTabBarWidget().currentIndex, 0); + + await widgetTester.sendKeyDownEvent(LogicalKeyboardKey.control); + + await widgetTester.sendKeyDownEvent(LogicalKeyboardKey.digit1); + await widgetTester.sendKeyUpEvent(LogicalKeyboardKey.digit1); + await widgetTester.pumpAndSettle(); + + expect(editableTabBarWidget().currentIndex, 0, + reason: 'Tab index should remain at 0'); + + await widgetTester.sendKeyDownEvent(LogicalKeyboardKey.digit2); + await widgetTester.sendKeyUpEvent(LogicalKeyboardKey.digit2); + await widgetTester.pumpAndSettle(); + + expect(editableTabBarWidget().currentIndex, 1); + + await widgetTester.sendKeyDownEvent(LogicalKeyboardKey.digit5); + await widgetTester.sendKeyUpEvent(LogicalKeyboardKey.digit5); + await widgetTester.pumpAndSettle(); + + expect(editableTabBarWidget().currentIndex, 1, + reason: + 'Tab index should remain at 1 since there is no tab at index 4'); + }); + testWidgets('Renaming tab', (widgetTester) async { FlutterError.onError = ignoreOverflowErrors; - setupMockOfflineNT4(); await widgetTester.pumpWidget( MaterialApp( home: DashboardPage( + ntConnection: createMockOfflineNT4(), preferences: preferences, version: '0.0.0.0', ), @@ -722,11 +979,11 @@ void main() { testWidgets('Duplicating tab', (widgetTester) async { FlutterError.onError = ignoreOverflowErrors; - setupMockOfflineNT4(); await widgetTester.pumpWidget( MaterialApp( home: DashboardPage( + ntConnection: createMockOfflineNT4(), preferences: preferences, version: '0.0.0.0', ), @@ -754,11 +1011,11 @@ void main() { testWidgets('Minimizing window', (widgetTester) async { FlutterError.onError = ignoreOverflowErrors; - setupMockOfflineNT4(); await widgetTester.pumpWidget( MaterialApp( home: DashboardPage( + ntConnection: createMockOfflineNT4(), preferences: preferences, version: '0.0.0.0', ), @@ -778,11 +1035,11 @@ void main() { testWidgets('Maximizing/unmaximizing window', (widgetTester) async { FlutterError.onError = ignoreOverflowErrors; - setupMockOfflineNT4(); await widgetTester.pumpWidget( MaterialApp( home: DashboardPage( + ntConnection: createMockOfflineNT4(), preferences: preferences, version: '0.0.0.0', ), @@ -810,11 +1067,11 @@ void main() { testWidgets('Closing window (All changes saved)', (widgetTester) async { FlutterError.onError = ignoreOverflowErrors; - setupMockOfflineNT4(); await widgetTester.pumpWidget( MaterialApp( home: DashboardPage( + ntConnection: createMockOfflineNT4(), preferences: preferences, version: '0.0.0.0', ), @@ -846,11 +1103,11 @@ void main() { testWidgets('Closing window (Unsaved changes)', (widgetTester) async { FlutterError.onError = ignoreOverflowErrors; - setupMockOfflineNT4(); await widgetTester.pumpWidget( MaterialApp( home: DashboardPage( + ntConnection: createMockOfflineNT4(), preferences: preferences, version: '0.0.0.0', ), @@ -888,11 +1145,11 @@ void main() { testWidgets('Opening settings', (widgetTester) async { FlutterError.onError = ignoreOverflowErrors; - setupMockOfflineNT4(); await widgetTester.pumpWidget( MaterialApp( home: DashboardPage( + ntConnection: createMockOfflineNT4(), preferences: preferences, version: '0.0.0.0', ), @@ -914,4 +1171,81 @@ void main() { expect(find.byType(SettingsDialog), findsOneWidget); }); + + testWidgets( + 'Robot Notifications', + (widgetTester) async { + FlutterError.onError = ignoreOverflowErrors; + final Map data = { + 'title': 'Robot Notification Title', + 'description': 'Robot Notification Description', + 'level': 'INFO' + }; + + MockNTConnection connection = createMockOnlineNT4(virtualTopics: [ + NT4Topic( + name: '/Elastic/RobotNotifications', + type: NT4TypeStr.kString, + properties: {}, + ) + ], virtualValues: { + '/Elastic/RobotNotifications': jsonEncode(data) + }); + MockNT4Subscription mockSub = MockNT4Subscription(); + + List listeners = []; + when(mockSub.listen(any)).thenAnswer( + (realInvocation) { + listeners.add(realInvocation.positionalArguments[0]); + mockSub.updateValue(jsonEncode(data), 0); + }, + ); + + when(mockSub.updateValue(any, any)).thenAnswer( + (invoc) { + for (var value in listeners) { + value.call( + invoc.positionalArguments[0], invoc.positionalArguments[1]); + } + }, + ); + + when(connection.subscribeAll(any, any)).thenAnswer( + (realInvocation) { + return mockSub; + }, + ); + + final notificationWidget = + find.widgetWithText(ElegantNotification, data['title']); + + await widgetTester.pumpWidget( + MaterialApp( + home: DashboardPage( + ntConnection: connection, + preferences: preferences, + version: '0.0.0.0', + ), + ), + ); + expect(notificationWidget, findsNothing); + + await widgetTester.pumpAndSettle(); + connection + .subscribeAll('/Elastic/robotnotifications', 0.2) + .updateValue(jsonEncode(data), 1); + + await widgetTester.pump(); + + expect(notificationWidget, findsOneWidget); + + await widgetTester.pumpAndSettle(); + + expect(notificationWidget, findsNothing); + + connection + .subscribeAll('/Elastic/robotnotifications', 0.2) + .updateValue(jsonEncode(data), 1); + }, + ); } diff --git a/test/services/hotkey_manager_test.dart b/test/services/hotkey_manager_test.dart new file mode 100644 index 00000000..8825d759 --- /dev/null +++ b/test/services/hotkey_manager_test.dart @@ -0,0 +1,72 @@ +import 'package:flutter/services.dart'; + +import 'package:flutter_test/flutter_test.dart'; +import 'package:mockito/mockito.dart'; + +import 'package:elastic_dashboard/services/hotkey_manager.dart'; + +class MockShortcutCallback extends Mock { + void callback(); +} + +void main() { + TestWidgetsFlutterBinding.ensureInitialized(); + + tearDown(() { + hotKeyManager.tearDown(); + HardwareKeyboard.instance.clearState(); + }); + + test('Shortcut no modifiers', () async { + MockShortcutCallback mockCallback = MockShortcutCallback(); + + hotKeyManager.register( + HotKey(LogicalKeyboardKey.keyA), + callback: mockCallback.callback, + ); + + await simulateKeyDownEvent(LogicalKeyboardKey.keyA); + + verifyNever(mockCallback.callback()); + + await simulateKeyUpEvent(LogicalKeyboardKey.keyA); + + verify(mockCallback.callback()).called(1); + + await simulateKeyDownEvent(LogicalKeyboardKey.control); + + await simulateKeyDownEvent(LogicalKeyboardKey.keyA); + await simulateKeyUpEvent(LogicalKeyboardKey.keyA); + + verify(mockCallback.callback()).called(1); + }); + + test('Shortcut with modifiers', () async { + MockShortcutCallback mockCallback = MockShortcutCallback(); + + hotKeyManager.register( + HotKey( + LogicalKeyboardKey.keyA, + modifiers: [KeyModifier.control, KeyModifier.shift, KeyModifier.alt], + ), + callback: mockCallback.callback, + ); + + await simulateKeyDownEvent(LogicalKeyboardKey.keyA); + + verifyNever(mockCallback.callback()); + + await simulateKeyUpEvent(LogicalKeyboardKey.keyA); + + verifyNever(mockCallback.callback()); + + await simulateKeyDownEvent(LogicalKeyboardKey.control); + await simulateKeyDownEvent(LogicalKeyboardKey.shift); + await simulateKeyDownEvent(LogicalKeyboardKey.alt); + + await simulateKeyDownEvent(LogicalKeyboardKey.keyA); + await simulateKeyUpEvent(LogicalKeyboardKey.keyA); + + verify(mockCallback.callback()).called(1); + }); +} diff --git a/test/services/nt_test.dart b/test/services/nt_test.dart index 7710761e..c7189ec6 100644 --- a/test/services/nt_test.dart +++ b/test/services/nt_test.dart @@ -1,61 +1,65 @@ import 'package:flutter_test/flutter_test.dart'; import 'package:elastic_dashboard/services/nt4_client.dart'; +import 'package:elastic_dashboard/services/nt_connection.dart'; void main() { test('NT4 Client', () { - bool connected = false; + NTConnection ntConnection = NTConnection('10.3.5.32'); - NT4Client client = NT4Client( - serverBaseAddress: '10.3.53.2', - onConnect: () => connected = true, - onDisconnect: () => connected = false, - ); - - expect(connected, false); + expect(ntConnection.isNT4Connected, false); // Subscribing NT4Subscription subscription1 = - client.subscribe('/SmartDashboard/Test Number'); + ntConnection.subscribe('/SmartDashboard/Test Number'); - expect(client.subscriptions.length, greaterThanOrEqualTo(1)); + expect(ntConnection.subscriptions.length, greaterThanOrEqualTo(1)); - expect(client.lastAnnouncedValues.isEmpty, true); + expect(ntConnection.getLastAnnouncedValue('/SmartDashboard/Test Number'), + isNull); // Publishing and adding to the last announced values - client.addSample( + ntConnection.updateDataFromSubscription(subscription1, 3.53); + + expect(ntConnection.getLastAnnouncedValue('/SmartDashboard/Test Number'), + isNull); + + ntConnection.updateDataFromTopic( NT4Topic( name: '/SmartDashboard/Test Number', type: NT4TypeStr.kFloat32, properties: {}), 3.53); - expect(client.lastAnnouncedValues.isEmpty, false); + expect(ntConnection.getLastAnnouncedValue('/SmartDashboard/Test Number'), + 3.53); expect(subscription1.currentValue != null, true); NT4Subscription subscription2 = - client.subscribe('/SmartDashboard/Test Number'); + ntConnection.subscribe('/SmartDashboard/Test Number'); // If the subscriptions are shared - expect(client.subscribedTopics.length, 1); + expect(ntConnection.subscriptions.length, 1); - client.unSubscribe(subscription1); + ntConnection.unSubscribe(subscription1); - expect(client.subscribedTopics.length, 1); + expect(ntConnection.subscriptions.length, 1); - client.unSubscribe(subscription2); + ntConnection.unSubscribe(subscription2); - expect(client.subscribedTopics.length, 0); + expect(ntConnection.subscriptions.length, 0); // Changing ip address - expect(client.lastAnnouncedValues.isEmpty, false); + expect(ntConnection.getLastAnnouncedValue('/SmartDashboard/Test Number'), + 3.53); - client.setServerBaseAddreess('10.26.01.2'); + ntConnection.changeIPAddress('10.30.15.2'); - expect(client.serverBaseAddress, '10.26.01.2'); + expect(ntConnection.serverBaseAddress, '10.30.15.2'); - expect(client.announcedTopics.isEmpty, true); - expect(client.lastAnnouncedValues.isEmpty, false); + expect(ntConnection.announcedTopics().length, 0); + expect(ntConnection.getLastAnnouncedValue('/SmartDashboard/Test Number'), + 3.53); }); } diff --git a/test/services/robot_notifications_listener_test.dart b/test/services/robot_notifications_listener_test.dart new file mode 100644 index 00000000..0c0905bc --- /dev/null +++ b/test/services/robot_notifications_listener_test.dart @@ -0,0 +1,114 @@ +import 'dart:convert'; + +import 'package:flutter/material.dart'; + +import 'package:flutter_test/flutter_test.dart'; +import 'package:mockito/mockito.dart'; + +import 'package:elastic_dashboard/services/robot_notifications_listener.dart'; +import '../test_util.dart'; +import '../test_util.mocks.dart'; + +class MockNotificationCallback extends Mock { + void call(String? title, String? description, Icon? icon); +} + +void main() { + test("Robot Notifications (Initial Connection | No Existing Data) ", () { + MockNTConnection mockConnection = createMockOnlineNT4(); + + // Create a mock for the onNotification callback + MockNotificationCallback mockOnNotification = MockNotificationCallback(); + + RobotNotificationsListener notifications = RobotNotificationsListener( + ntConnection: mockConnection, + onNotification: mockOnNotification.call, + ); + + notifications.listen(); + + // Verify that subscribeAll was called with the specific parameters + verify(mockConnection.subscribeAll('/Elastic/robotnotifications', 0.2)) + .called(1); + verify(mockConnection.addDisconnectedListener(any)).called(1); + + // Verify that no other interactions have been made with the mockConnection + verifyNoMoreInteractions(mockConnection); + + // Verify that the onNotification callback was never called + verifyNever(mockOnNotification.call(any, any, any)); + }); + + test("Robot Notifications (Initial Connection | Existing Data) ", () { + MockNTConnection mockConnection = createMockOnlineNT4(); + MockNT4Subscription mockSub = MockNT4Subscription(); + + Map data = { + 'title': 'Title1', + 'description': 'Description1', + 'level': 'Info' + }; + + List listeners = []; + when(mockSub.listen(any)).thenAnswer( + (realInvocation) { + listeners.add(realInvocation.positionalArguments[0]); + mockSub.updateValue(jsonEncode(data), 0); + }, + ); + + when(mockSub.updateValue(any, any)).thenAnswer( + (invoc) { + for (var value in listeners) { + value.call( + invoc.positionalArguments[0], invoc.positionalArguments[1]); + } + }, + ); + + when(mockConnection.subscribeAll(any, any)).thenAnswer( + (realInvocation) { + mockSub.updateValue(jsonEncode(data), 0); + return mockSub; + }, + ); + + // Create a mock for the onNotification callback + MockNotificationCallback mockOnNotification = MockNotificationCallback(); + + RobotNotificationsListener notifications = RobotNotificationsListener( + ntConnection: mockConnection, + onNotification: mockOnNotification.call, + ); + + notifications.listen(); + + // Verify that subscribeAll was called with the specific parameters + verify(mockConnection.subscribeAll('/Elastic/robotnotifications', 0.2)) + .called(1); + verify(mockConnection.addDisconnectedListener(any)).called(1); + + // Verify that no other interactions have been made with the mockConnection + verifyNoMoreInteractions(mockConnection); + + // Verify that the onNotification callback was never called + verifyNever(mockOnNotification(any, any, any)); + + // Publish some data and expect an update + data['title'] = 'Title2'; + data['description'] = 'Description2'; + data['level'] = 'INFO'; + mockSub.updateValue(jsonEncode(data), 2); + + verify(mockOnNotification(data['title'], data['description'], any)); + + // Try malformed data + data['title'] = null; + data['description'] = null; + data['level'] = 'malformedlevel'; + + mockSub.updateValue(jsonEncode(data), 3); + reset(mockOnNotification); + verifyNever(mockOnNotification(any, any, any)); + }); +} diff --git a/test/services/shuffleboard_nt_listener_test.dart b/test/services/shuffleboard_nt_listener_test.dart index 816af343..9891c490 100644 --- a/test/services/shuffleboard_nt_listener_test.dart +++ b/test/services/shuffleboard_nt_listener_test.dart @@ -1,13 +1,20 @@ import 'package:flutter_test/flutter_test.dart'; import 'package:mockito/mockito.dart'; +import 'package:shared_preferences/shared_preferences.dart'; import 'package:elastic_dashboard/services/nt4_client.dart'; -import 'package:elastic_dashboard/services/nt_connection.dart'; import 'package:elastic_dashboard/services/settings.dart'; import 'package:elastic_dashboard/services/shuffleboard_nt_listener.dart'; import '../test_util.mocks.dart'; void main() { + late SharedPreferences preferences; + + setUp(() async { + SharedPreferences.setMockInitialValues({}); + preferences = await SharedPreferences.getInstance(); + }); + test('Shuffleboard NT listener', () async { List topicAnnounceListeners = []; Map lastAnnouncedValues = { @@ -16,21 +23,15 @@ void main() { }; final mockNT4Connection = MockNTConnection(); - final mockNT4Client = MockNT4Client(); final mockSubscription = MockNT4Subscription(); - when(mockNT4Client.lastAnnouncedValues).thenReturn(lastAnnouncedValues); - when(mockNT4Client.topicAnnounceListeners) - .thenReturn(topicAnnounceListeners); - when(mockNT4Client.addTopicAnnounceListener(any)).thenAnswer( + when(mockNT4Connection.addTopicAnnounceListener(any)).thenAnswer( (realInvocation) => topicAnnounceListeners.add(realInvocation.positionalArguments[0])); when(mockSubscription.periodicStream()) .thenAnswer((_) => Stream.value(null)); - when(mockNT4Connection.nt4Client).thenReturn(mockNT4Client); - when(mockNT4Connection.isNT4Connected).thenReturn(true); when(mockNT4Connection.latencyStream()).thenAnswer((_) => Stream.value(0)); @@ -48,15 +49,13 @@ void main() { when(mockNT4Connection.subscribe(any)).thenReturn(mockSubscription); - when(mockNT4Connection.subscribeAll(any, any)).thenReturn(mockSubscription); - - when(mockNT4Connection.subscribeAll(any)).thenReturn(mockSubscription); - - NTConnection.instance = mockNT4Connection; + // NTConnection.instance = mockNT4Connection; Map announcedWidgetData = {}; ShuffleboardNTListener ntListener = ShuffleboardNTListener( + ntConnection: mockNT4Connection, + preferences: preferences, onWidgetAdded: (widgetData) { widgetData.forEach( (key, value) => announcedWidgetData.putIfAbsent(key, () => value)); @@ -65,9 +64,9 @@ void main() { ..initializeSubscriptions() ..initializeListeners(); - expect(ntConnection.nt4Client.topicAnnounceListeners.isNotEmpty, true); + expect(topicAnnounceListeners.isNotEmpty, true); - for (final callback in ntConnection.nt4Client.topicAnnounceListeners) { + for (final callback in topicAnnounceListeners) { callback.call(NT4Topic( name: '/Shuffleboard/.metadata/Test-Tab/Test Number/Position', type: NT4TypeStr.kFloat32Arr, @@ -97,9 +96,9 @@ void main() { expect(announcedWidgetData.containsKey('width'), true); expect(announcedWidgetData.containsKey('height'), true); - expect(announcedWidgetData['x'], Settings.gridSize.toDouble()); - expect(announcedWidgetData['y'], Settings.gridSize.toDouble()); - expect(announcedWidgetData['width'], Settings.gridSize.toDouble() * 2.0); - expect(announcedWidgetData['height'], Settings.gridSize.toDouble() * 2.0); + expect(announcedWidgetData['x'], Defaults.gridSize.toDouble()); + expect(announcedWidgetData['y'], Defaults.gridSize.toDouble()); + expect(announcedWidgetData['width'], Defaults.gridSize.toDouble() * 2.0); + expect(announcedWidgetData['height'], Defaults.gridSize.toDouble() * 2.0); }); } diff --git a/test/test_util.dart b/test/test_util.dart index 9428e11a..891b39b8 100644 --- a/test/test_util.dart +++ b/test/test_util.dart @@ -14,16 +14,17 @@ import 'test_util.mocks.dart'; MockSpec(), MockSpec() ]) -void setupMockOfflineNT4() { +MockNTConnection createMockOfflineNT4() { HttpOverrides.global = null; final mockNT4Connection = MockNTConnection(); - final mockNT4Client = MockNT4Client(); final mockSubscription = MockNT4Subscription(); + when(mockNT4Connection.announcedTopics()).thenReturn({}); + when(mockSubscription.periodicStream()).thenAnswer((_) => Stream.value(null)); - when(mockNT4Connection.nt4Client).thenReturn(mockNT4Client); + when(mockSubscription.listen(any)).thenAnswer((realInvocation) {}); when(mockNT4Connection.isNT4Connected).thenReturn(false); @@ -43,37 +44,48 @@ void setupMockOfflineNT4() { when(mockNT4Connection.subscribeAll(any, any)).thenReturn(mockSubscription); - when(mockNT4Connection.subscribeAll(any)).thenReturn(mockSubscription); - - when(mockNT4Connection.getTopicFromName(any)) - .thenReturn(NT4Topic(name: '', type: NT4TypeStr.kString, properties: {})); + when(mockNT4Connection.getTopicFromName(any)).thenReturn(null); - NTConnection.instance = mockNT4Connection; + return mockNT4Connection; } -void setupMockOnlineNT4() { +MockNTConnection createMockOnlineNT4({ + List? virtualTopics, + Map? virtualValues, +}) { HttpOverrides.global = null; final mockNT4Connection = MockNTConnection(); - final mockNT4Client = MockNT4Client(); final mockSubscription = MockNT4Subscription(); - when(mockNT4Client.announcedTopics).thenReturn({ - 1: NT4Topic( + virtualTopics ??= [ + NT4Topic( name: '/SmartDashboard/Test Value 1', type: NT4TypeStr.kInt, properties: {}, ), - 2: NT4Topic( + NT4Topic( name: '/SmartDashboard/Test Value 2', type: NT4TypeStr.kFloat32, properties: {}, ), - }); + ]; + + virtualValues ??= {}; + + Map virtualTopicsMap = {}; + + List subscriptionListeners = []; + + for (int i = 0; i < virtualTopics.length; i++) { + virtualTopicsMap.addAll({i + 1: virtualTopics[i]}); + } + + when(mockNT4Connection.announcedTopics()).thenReturn(virtualTopicsMap); when(mockSubscription.periodicStream()).thenAnswer((_) => Stream.value(null)); - when(mockNT4Connection.nt4Client).thenReturn(mockNT4Client); + when(mockSubscription.listen(any)).thenAnswer((realInvocation) {}); when(mockNT4Connection.isNT4Connected).thenReturn(true); @@ -93,12 +105,66 @@ void setupMockOnlineNT4() { when(mockNT4Connection.subscribeAll(any, any)).thenReturn(mockSubscription); - when(mockNT4Connection.subscribeAll(any)).thenReturn(mockSubscription); + when(mockNT4Connection.getTopicFromName(any)).thenReturn(null); + + when(mockNT4Connection.publishNewTopic(any, any)).thenAnswer((invocation) { + NT4Topic newTopic = NT4Topic( + name: invocation.positionalArguments[0], + type: invocation.positionalArguments[1], + properties: {}); + + virtualTopicsMap[virtualTopicsMap.length] = newTopic; + return newTopic; + }); + + when(mockNT4Connection.updateDataFromTopic(any, any)) + .thenAnswer((invocation) { + NT4Topic topic = invocation.positionalArguments[0]; + Object? data = invocation.positionalArguments[1]; - when(mockNT4Connection.getTopicFromName(any)) - .thenReturn(NT4Topic(name: '', type: NT4TypeStr.kString, properties: {})); + virtualValues![topic.name] = data; + }); + + when(mockNT4Connection.updateDataFromTopicName(any, any)) + .thenAnswer((invocation) { + String topic = invocation.positionalArguments[0]; + Object? data = invocation.positionalArguments[1]; + + virtualValues![topic] = data; + }); + + for (NT4Topic topic in virtualTopics) { + MockNT4Subscription topicSubscription = MockNT4Subscription(); + + when(mockNT4Connection.getTopicFromName(topic.name)).thenReturn(topic); + + when(topicSubscription.periodicStream(yieldAll: anyNamed('yieldAll'))) + .thenAnswer((_) => Stream.value(virtualValues?[topic.name])); + + when(topicSubscription.listen(any)).thenAnswer((realInvocation) { + subscriptionListeners.add(realInvocation.positionalArguments[0]); + }); + + when(topicSubscription.updateValue(any, any)).thenAnswer( + (invoc) { + for (var value in subscriptionListeners) { + value.call( + invoc.positionalArguments[0], invoc.positionalArguments[1]); + } + }, + ); + + when(mockNT4Connection.getLastAnnouncedValue(topic.name)) + .thenAnswer((_) => virtualValues?[topic.name]); + + when(mockNT4Connection.subscribe(topic.name, any)) + .thenReturn(topicSubscription); + + when(mockNT4Connection.subscribeAll(topic.name, any)) + .thenReturn(topicSubscription); + } - NTConnection.instance = mockNT4Connection; + return mockNT4Connection; } void ignoreOverflowErrors( diff --git a/test/widgets/editable_tab_bar_test.dart b/test/widgets/editable_tab_bar_test.dart index e80379cd..fe8585fa 100644 --- a/test/widgets/editable_tab_bar_test.dart +++ b/test/widgets/editable_tab_bar_test.dart @@ -4,11 +4,14 @@ import 'package:flutter/material.dart'; import 'package:flutter_test/flutter_test.dart'; import 'package:mockito/annotations.dart'; import 'package:mockito/mockito.dart'; +import 'package:shared_preferences/shared_preferences.dart'; +import 'package:elastic_dashboard/util/tab_data.dart'; import 'package:elastic_dashboard/widgets/dialog_widgets/dialog_text_input.dart'; import 'package:elastic_dashboard/widgets/editable_tab_bar.dart'; import 'package:elastic_dashboard/widgets/tab_grid.dart'; import '../test_util.dart'; +import '../test_util.mocks.dart'; import 'editable_tab_bar_test.mocks.dart'; @GenerateNiceMocks([MockSpec()]) @@ -30,9 +33,14 @@ class FakeTabBarFunctions { void main() { late MockFakeTabBarFunctions tabBarFunctions; + late MockNTConnection mockNTConnection; + late SharedPreferences preferences; - setUp(() { + setUp(() async { tabBarFunctions = MockFakeTabBarFunctions(); + SharedPreferences.setMockInitialValues({}); + preferences = await SharedPreferences.getInstance(); + mockNTConnection = createMockOfflineNT4(); }); testWidgets('Editable tab bar', (widgetTester) async { @@ -42,22 +50,34 @@ void main() { MaterialApp( home: Scaffold( body: EditableTabBar( - currentIndex: 0, - tabData: [ - TabData(name: 'Teleoperated'), - TabData(name: 'Autonomous'), - ], - tabViews: [ - TabGrid(onAddWidgetPressed: () {}), - TabGrid(onAddWidgetPressed: () {}), - ], - onTabCreate: (tab) {}, - onTabDestroy: (index) {}, - onTabMoveLeft: () {}, - onTabMoveRight: () {}, - onTabRename: (tab, grid) {}, - onTabChanged: (index) {}, - onTabDuplicate: (index, newData) {}), + preferences: preferences, + currentIndex: 0, + tabData: [ + TabData( + name: 'Teleoperated', + tabGrid: TabGridModel( + ntConnection: mockNTConnection, + preferences: preferences, + onAddWidgetPressed: () {}, + ), + ), + TabData( + name: 'Autonomous', + tabGrid: TabGridModel( + ntConnection: mockNTConnection, + preferences: preferences, + onAddWidgetPressed: () {}, + ), + ), + ], + onTabCreate: () {}, + onTabDestroy: (index) {}, + onTabMoveLeft: () {}, + onTabMoveRight: () {}, + onTabRename: (tab, grid) {}, + onTabChanged: (index) {}, + onTabDuplicate: (index) {}, + ), ), ), ); @@ -82,16 +102,27 @@ void main() { MaterialApp( home: Scaffold( body: EditableTabBar( + preferences: preferences, currentIndex: 0, tabData: [ - TabData(name: 'Teleoperated'), - TabData(name: 'Autonomous'), - ], - tabViews: [ - TabGrid(onAddWidgetPressed: () {}), - TabGrid(onAddWidgetPressed: () {}), + TabData( + name: 'Teleoperated', + tabGrid: TabGridModel( + ntConnection: mockNTConnection, + preferences: preferences, + onAddWidgetPressed: () {}, + ), + ), + TabData( + name: 'Autonomous', + tabGrid: TabGridModel( + ntConnection: mockNTConnection, + preferences: preferences, + onAddWidgetPressed: () {}, + ), + ), ], - onTabCreate: (tab) { + onTabCreate: () { tabBarFunctions.onTabCreate(); }, onTabDestroy: (index) { @@ -109,7 +140,7 @@ void main() { onTabChanged: (index) { tabBarFunctions.onTabChanged(); }, - onTabDuplicate: (index, tab) { + onTabDuplicate: (index) { tabBarFunctions.onTabDuplicate(); }, ), @@ -136,16 +167,27 @@ void main() { MaterialApp( home: Scaffold( body: EditableTabBar( + preferences: preferences, currentIndex: 0, tabData: [ - TabData(name: 'Teleoperated'), - TabData(name: 'Autonomous'), + TabData( + name: 'Teleoperated', + tabGrid: TabGridModel( + ntConnection: mockNTConnection, + preferences: preferences, + onAddWidgetPressed: () {}, + ), + ), + TabData( + name: 'Autonomous', + tabGrid: TabGridModel( + ntConnection: mockNTConnection, + preferences: preferences, + onAddWidgetPressed: () {}, + ), + ), ], - tabViews: [ - TabGrid(onAddWidgetPressed: () {}), - TabGrid(onAddWidgetPressed: () {}), - ], - onTabCreate: (tab) { + onTabCreate: () { tabBarFunctions.onTabCreate(); }, onTabDestroy: (index) { @@ -163,7 +205,7 @@ void main() { onTabChanged: (index) { tabBarFunctions.onTabChanged(); }, - onTabDuplicate: (index, tab) { + onTabDuplicate: (index) { tabBarFunctions.onTabDuplicate(); }, ), @@ -190,16 +232,27 @@ void main() { MaterialApp( home: Scaffold( body: EditableTabBar( + preferences: preferences, currentIndex: 0, tabData: [ - TabData(name: 'Teleoperated'), - TabData(name: 'Autonomous'), - ], - tabViews: [ - TabGrid(onAddWidgetPressed: () {}), - TabGrid(onAddWidgetPressed: () {}), + TabData( + name: 'Teleoperated', + tabGrid: TabGridModel( + ntConnection: mockNTConnection, + preferences: preferences, + onAddWidgetPressed: () {}, + ), + ), + TabData( + name: 'Autonomous', + tabGrid: TabGridModel( + ntConnection: mockNTConnection, + preferences: preferences, + onAddWidgetPressed: () {}, + ), + ), ], - onTabCreate: (tab) { + onTabCreate: () { tabBarFunctions.onTabCreate(); }, onTabDestroy: (index) { @@ -217,7 +270,7 @@ void main() { onTabChanged: (index) { tabBarFunctions.onTabChanged(); }, - onTabDuplicate: (index, tab) { + onTabDuplicate: (index) { tabBarFunctions.onTabDuplicate(); }, ), @@ -257,16 +310,27 @@ void main() { MaterialApp( home: Scaffold( body: EditableTabBar( + preferences: preferences, currentIndex: 0, tabData: [ - TabData(name: 'Teleoperated'), - TabData(name: 'Autonomous'), + TabData( + name: 'Teleoperated', + tabGrid: TabGridModel( + ntConnection: mockNTConnection, + preferences: preferences, + onAddWidgetPressed: () {}, + ), + ), + TabData( + name: 'Autonomous', + tabGrid: TabGridModel( + ntConnection: mockNTConnection, + preferences: preferences, + onAddWidgetPressed: () {}, + ), + ), ], - tabViews: [ - TabGrid(onAddWidgetPressed: () {}), - TabGrid(onAddWidgetPressed: () {}), - ], - onTabCreate: (tab) { + onTabCreate: () { tabBarFunctions.onTabCreate(); }, onTabDestroy: (index) { @@ -284,7 +348,7 @@ void main() { onTabChanged: (index) { tabBarFunctions.onTabChanged(); }, - onTabDuplicate: (index, tab) { + onTabDuplicate: (index) { tabBarFunctions.onTabDuplicate(); }, ), @@ -335,16 +399,27 @@ void main() { MaterialApp( home: Scaffold( body: EditableTabBar( + preferences: preferences, currentIndex: 0, tabData: [ - TabData(name: 'Teleoperated'), - TabData(name: 'Autonomous'), - ], - tabViews: [ - TabGrid(onAddWidgetPressed: () {}), - TabGrid(onAddWidgetPressed: () {}), + TabData( + name: 'Teleoperated', + tabGrid: TabGridModel( + ntConnection: mockNTConnection, + preferences: preferences, + onAddWidgetPressed: () {}, + ), + ), + TabData( + name: 'Autonomous', + tabGrid: TabGridModel( + ntConnection: mockNTConnection, + preferences: preferences, + onAddWidgetPressed: () {}, + ), + ), ], - onTabCreate: (tab) { + onTabCreate: () { tabBarFunctions.onTabCreate(); }, onTabDestroy: (index) { @@ -362,7 +437,7 @@ void main() { onTabChanged: (index) { tabBarFunctions.onTabChanged(); }, - onTabDuplicate: (index, tab) { + onTabDuplicate: (index) { tabBarFunctions.onTabDuplicate(); }, ), diff --git a/test/widgets/nt_widgets/multi-topic/accelerometer_test.dart b/test/widgets/nt_widgets/multi-topic/accelerometer_test.dart new file mode 100644 index 00000000..61ae3736 --- /dev/null +++ b/test/widgets/nt_widgets/multi-topic/accelerometer_test.dart @@ -0,0 +1,94 @@ +import 'package:flutter/material.dart'; + +import 'package:flutter_test/flutter_test.dart'; +import 'package:provider/provider.dart'; +import 'package:shared_preferences/shared_preferences.dart'; + +import 'package:elastic_dashboard/services/nt4_client.dart'; +import 'package:elastic_dashboard/services/nt_connection.dart'; +import 'package:elastic_dashboard/services/nt_widget_builder.dart'; +import 'package:elastic_dashboard/widgets/nt_widgets/multi-topic/accelerometer.dart'; +import 'package:elastic_dashboard/widgets/nt_widgets/nt_widget.dart'; +import '../../../test_util.dart'; + +void main() { + TestWidgetsFlutterBinding.ensureInitialized(); + + final Map accelerometerJson = { + 'topic': 'Test/Test Accelerometer', + 'period': 0.100, + }; + + late SharedPreferences preferences; + late NTConnection ntConnection; + + setUp(() async { + SharedPreferences.setMockInitialValues({}); + preferences = await SharedPreferences.getInstance(); + + ntConnection = createMockOnlineNT4( + virtualTopics: [ + NT4Topic( + name: 'Test/Test Accelerometer/Value', + type: NT4TypeStr.kFloat32, + properties: {}, + ) + ], + virtualValues: { + 'Test/Test Accelerometer/Value': 0.50, + }, + ); + }); + + test('Creating accelerometer from json', () { + NTWidgetModel accelerometerModel = NTWidgetBuilder.buildNTModelFromJson( + ntConnection, + preferences, + 'Accelerometer', + accelerometerJson, + ); + + expect(accelerometerModel.type, 'Accelerometer'); + expect(accelerometerModel.runtimeType, AccelerometerModel); + + expect((accelerometerModel as AccelerometerModel).valueTopic, + 'Test/Test Accelerometer/Value'); + }); + + test('Saving accelerometer to json', () { + AccelerometerModel accelerometerModel = AccelerometerModel( + ntConnection: ntConnection, + preferences: preferences, + topic: 'Test/Test Accelerometer', + period: 0.100, + ); + + expect(accelerometerModel.toJson(), accelerometerJson); + }); + + testWidgets('Accelerometer widget test', (widgetTester) async { + FlutterError.onError = ignoreOverflowErrors; + + NTWidgetModel accelerometerModel = NTWidgetBuilder.buildNTModelFromJson( + ntConnection, + preferences, + 'Accelerometer', + accelerometerJson, + ); + + await widgetTester.pumpWidget( + MaterialApp( + home: Scaffold( + body: ChangeNotifierProvider.value( + value: accelerometerModel, + child: const AccelerometerWidget(), + ), + ), + ), + ); + + await widgetTester.pumpAndSettle(); + + expect(find.text('0.50 g'), findsOneWidget); + }); +} diff --git a/test/widgets/nt_widgets/multi-topic/basic_swerve_drive_test.dart b/test/widgets/nt_widgets/multi-topic/basic_swerve_drive_test.dart new file mode 100644 index 00000000..8bdd0759 --- /dev/null +++ b/test/widgets/nt_widgets/multi-topic/basic_swerve_drive_test.dart @@ -0,0 +1,106 @@ +import 'package:flutter/material.dart'; + +import 'package:flutter_test/flutter_test.dart'; +import 'package:provider/provider.dart'; +import 'package:shared_preferences/shared_preferences.dart'; + +import 'package:elastic_dashboard/services/nt_connection.dart'; +import 'package:elastic_dashboard/services/nt_widget_builder.dart'; +import 'package:elastic_dashboard/widgets/nt_widgets/multi-topic/basic_swerve_drive.dart'; +import 'package:elastic_dashboard/widgets/nt_widgets/nt_widget.dart'; +import '../../../test_util.dart'; + +void main() { + TestWidgetsFlutterBinding.ensureInitialized(); + + late SharedPreferences preferences; + late NTConnection ntConnection; + + final Map swerveJson = { + 'topic': 'Test/Basic Swerve Drive', + 'period': 0.100, + 'show_robot_rotation': false, + 'rotation_unit': 'Radians', + }; + + setUp(() async { + SharedPreferences.setMockInitialValues({}); + preferences = await SharedPreferences.getInstance(); + + ntConnection = createMockOnlineNT4(); + }); + + test('Basic swerve model from json', () { + NTWidgetModel swerveModel = NTWidgetBuilder.buildNTModelFromJson( + ntConnection, + preferences, + 'SwerveDrive', + swerveJson, + ); + + expect(swerveModel.type, 'SwerveDrive'); + expect(swerveModel.runtimeType, BasicSwerveModel); + + if (swerveModel is! BasicSwerveModel) { + return; + } + + expect(swerveModel.showRobotRotation, isFalse); + expect(swerveModel.rotationUnit, 'Radians'); + + expect(swerveModel.frontLeftAngleTopic, + 'Test/Basic Swerve Drive/Front Left Angle'); + expect(swerveModel.frontLeftVelocityTopic, + 'Test/Basic Swerve Drive/Front Left Velocity'); + + expect(swerveModel.frontRightAngleTopic, + 'Test/Basic Swerve Drive/Front Right Angle'); + expect(swerveModel.frontRightVelocityTopic, + 'Test/Basic Swerve Drive/Front Right Velocity'); + + expect(swerveModel.backLeftAngleTopic, + 'Test/Basic Swerve Drive/Back Left Angle'); + expect(swerveModel.backLeftVelocityTopic, + 'Test/Basic Swerve Drive/Back Left Velocity'); + + expect(swerveModel.backRightAngleTopic, + 'Test/Basic Swerve Drive/Back Right Angle'); + expect(swerveModel.backRightVelocityTopic, + 'Test/Basic Swerve Drive/Back Right Velocity'); + }); + + test('Basic swerve model to json', () { + BasicSwerveModel swerveModel = BasicSwerveModel( + ntConnection: ntConnection, + preferences: preferences, + topic: 'Test/Basic Swerve Drive', + period: 0.100, + rotationUnit: 'Radians', + showRobotRotation: false, + ); + + expect(swerveModel.toJson(), swerveJson); + }); + + testWidgets('Basic swerve widget test', (widgetTester) async { + FlutterError.onError = ignoreOverflowErrors; + + NTWidgetModel swerveModel = NTWidgetBuilder.buildNTModelFromJson( + ntConnection, preferences, 'SwerveDrive', swerveJson); + + await widgetTester.pumpWidget( + MaterialApp( + home: Scaffold( + body: ChangeNotifierProvider.value( + value: swerveModel, + child: const SwerveDriveWidget(), + ), + ), + ), + ); + + await widgetTester.pumpAndSettle(); + + expect(find.byType(CustomPaint), findsWidgets); + }); +} diff --git a/test/widgets/nt_widgets/multi-topic/camera_stream_test.dart b/test/widgets/nt_widgets/multi-topic/camera_stream_test.dart new file mode 100644 index 00000000..b5ac84ca --- /dev/null +++ b/test/widgets/nt_widgets/multi-topic/camera_stream_test.dart @@ -0,0 +1,141 @@ +import 'package:flutter/material.dart'; + +import 'package:flutter_test/flutter_test.dart'; +import 'package:provider/provider.dart'; +import 'package:shared_preferences/shared_preferences.dart'; + +import 'package:elastic_dashboard/services/nt_connection.dart'; +import 'package:elastic_dashboard/services/nt_widget_builder.dart'; +import 'package:elastic_dashboard/widgets/custom_loading_indicator.dart'; +import 'package:elastic_dashboard/widgets/nt_widgets/multi-topic/camera_stream.dart'; +import 'package:elastic_dashboard/widgets/nt_widgets/nt_widget.dart'; +import '../../../test_util.dart'; + +void main() { + TestWidgetsFlutterBinding.ensureInitialized(); + + final Map cameraStreamJson = { + 'topic': 'Test/Camera Stream', + 'period': 0.100, + 'compression': 50, + 'fps': 60, + 'resolution': [100.0, 100.0], + }; + + late SharedPreferences preferences; + late NTConnection ntConnection; + + setUp(() async { + SharedPreferences.setMockInitialValues({}); + preferences = await SharedPreferences.getInstance(); + + ntConnection = createMockOnlineNT4(); + }); + + test('Camera stream from json', () { + NTWidgetModel cameraStreamModel = NTWidgetBuilder.buildNTModelFromJson( + ntConnection, + preferences, + 'Camera Stream', + cameraStreamJson, + ); + + expect(cameraStreamModel.type, 'Camera Stream'); + expect(cameraStreamModel.runtimeType, CameraStreamModel); + + if (cameraStreamModel is! CameraStreamModel) { + return; + } + + expect(cameraStreamModel.fps, 60); + expect(cameraStreamModel.quality, 50); + expect(cameraStreamModel.resolution, const Size(100.0, 100.0)); + + expect(cameraStreamModel.getUrlWithParameters('0.0.0.0'), + '0.0.0.0?resolution=100x100&fps=60&compression=50'); + + cameraStreamModel.fps = null; + + expect(cameraStreamModel.getUrlWithParameters('0.0.0.0'), + '0.0.0.0?resolution=100x100&compression=50'); + + cameraStreamModel.resolution = const Size(0.0, 100); + + expect(cameraStreamModel.getUrlWithParameters('0.0.0.0'), + '0.0.0.0?compression=50'); + + cameraStreamModel.quality = null; + + expect(cameraStreamModel.getUrlWithParameters('0.0.0.0'), '0.0.0.0?'); + }); + + test('Camera stream to json', () { + CameraStreamModel cameraStreamModel = CameraStreamModel( + ntConnection: ntConnection, + preferences: preferences, + topic: 'Test/Camera Stream', + period: 0.100, + compression: 50, + fps: 60, + resolution: const Size(100.0, 100.0), + ); + + expect(cameraStreamModel.toJson(), cameraStreamJson); + }); + + testWidgets('Camera stream online widget test', (widgetTester) async { + FlutterError.onError = ignoreOverflowErrors; + + NTWidgetModel cameraStreamModel = NTWidgetBuilder.buildNTModelFromJson( + ntConnection, + preferences, + 'Camera Stream', + cameraStreamJson, + ); + + await widgetTester.pumpWidget( + MaterialApp( + home: Scaffold( + body: ChangeNotifierProvider.value( + value: cameraStreamModel, + child: const CameraStreamWidget(), + ), + ), + ), + ); + + await widgetTester.pumpAndSettle(); + + expect(find.byType(CustomLoadingIndicator), findsOneWidget); + expect( + find.text('Waiting for Camera Stream connection...'), findsOneWidget); + }); + + testWidgets('Camera stream offline widget test', (widgetTester) async { + FlutterError.onError = ignoreOverflowErrors; + + NTWidgetModel cameraStreamModel = NTWidgetBuilder.buildNTModelFromJson( + createMockOfflineNT4(), + preferences, + 'Camera Stream', + cameraStreamJson, + ); + + await widgetTester.pumpWidget( + MaterialApp( + home: Scaffold( + body: ChangeNotifierProvider.value( + value: cameraStreamModel, + child: const CameraStreamWidget(), + ), + ), + ), + ); + + await widgetTester.pumpAndSettle(); + + expect(find.byType(CustomLoadingIndicator), findsOneWidget); + expect( + find.text('Waiting for Network Tables connection...'), findsOneWidget); + }); +} diff --git a/test/widgets/nt_widgets/multi-topic/combo_box_chooser_test.dart b/test/widgets/nt_widgets/multi-topic/combo_box_chooser_test.dart new file mode 100644 index 00000000..1632988b --- /dev/null +++ b/test/widgets/nt_widgets/multi-topic/combo_box_chooser_test.dart @@ -0,0 +1,162 @@ +import 'package:flutter/material.dart'; + +import 'package:dropdown_button2/dropdown_button2.dart'; +import 'package:flutter_test/flutter_test.dart'; +import 'package:provider/provider.dart'; +import 'package:shared_preferences/shared_preferences.dart'; + +import 'package:elastic_dashboard/services/nt4_client.dart'; +import 'package:elastic_dashboard/services/nt_connection.dart'; +import 'package:elastic_dashboard/services/nt_widget_builder.dart'; +import 'package:elastic_dashboard/widgets/nt_widgets/multi-topic/combo_box_chooser.dart'; +import 'package:elastic_dashboard/widgets/nt_widgets/nt_widget.dart'; +import '../../../test_util.dart'; + +void main() { + TestWidgetsFlutterBinding.ensureInitialized(); + + final Map comboBoxChooserJson = { + 'topic': 'Test/Combo Box Chooser', + 'period': 0.100, + 'sort_options': true, + }; + + late SharedPreferences preferences; + late NTConnection ntConnection; + + setUp(() async { + SharedPreferences.setMockInitialValues({}); + preferences = await SharedPreferences.getInstance(); + + ntConnection = createMockOnlineNT4( + virtualTopics: [ + NT4Topic( + name: 'Test/Combo Box Chooser/options', + type: NT4TypeStr.kStringArr, + properties: {}), + NT4Topic( + name: 'Test/Combo Box Chooser/active', + type: NT4TypeStr.kString, + properties: {}), + NT4Topic( + name: 'Test/Combo Box Chooser/selected', + type: NT4TypeStr.kString, + properties: {}), + NT4Topic( + name: 'Test/Combo Box Chooser/default', + type: NT4TypeStr.kString, + properties: {}), + ], + virtualValues: { + 'Test/Combo Box Chooser/options': ['One', 'Two', 'Three'], + 'Test/Combo Box Chooser/active': 'Two', + }, + ); + }); + + test('Combo box chooser from json', () { + NTWidgetModel comboBoxChooserModel = NTWidgetBuilder.buildNTModelFromJson( + ntConnection, + preferences, + 'ComboBox Chooser', + comboBoxChooserJson, + ); + + expect(comboBoxChooserModel.type, 'ComboBox Chooser'); + expect(comboBoxChooserModel.runtimeType, ComboBoxChooserModel); + + if (comboBoxChooserModel is! ComboBoxChooserModel) { + return; + } + + expect(comboBoxChooserModel.sortOptions, isTrue); + }); + + test('Combo box chooser alias name', () { + NTWidgetModel comboBoxChooserModel = NTWidgetBuilder.buildNTModelFromJson( + ntConnection, + preferences, + 'String Chooser', + comboBoxChooserJson, + ); + + expect(comboBoxChooserModel.type, 'ComboBox Chooser'); + expect(comboBoxChooserModel.runtimeType, ComboBoxChooserModel); + + if (comboBoxChooserModel is! ComboBoxChooserModel) { + return; + } + + expect(comboBoxChooserModel.sortOptions, isTrue); + }); + + test('Combo box chooser to json', () { + ComboBoxChooserModel comboBoxChooserModel = ComboBoxChooserModel( + ntConnection: ntConnection, + preferences: preferences, + topic: 'Test/Combo Box Chooser', + period: 0.100, + sortOptions: true, + ); + + expect(comboBoxChooserModel.toJson(), comboBoxChooserJson); + }); + + testWidgets('Combo box chooser widget test', (widgetTester) async { + FlutterError.onError = ignoreOverflowErrors; + + NTWidgetModel comboBoxChooserModel = NTWidgetBuilder.buildNTModelFromJson( + ntConnection, + preferences, + 'ComboBox Chooser', + comboBoxChooserJson, + ); + + await widgetTester.pumpWidget( + MaterialApp( + home: Scaffold( + body: ChangeNotifierProvider.value( + value: comboBoxChooserModel, + child: const ComboBoxChooser(), + ), + ), + ), + ); + + await widgetTester.pumpAndSettle(); + + expect(find.byType(DropdownButton2), findsOneWidget); + expect(find.text('One'), findsNothing); + expect(find.text('Two'), findsOneWidget); + expect(find.text('Three'), findsNothing); + expect( + (comboBoxChooserModel as ComboBoxChooserModel).selectedChoice, 'Two'); + expect(find.byIcon(Icons.check), findsOneWidget); + + await widgetTester.tap(find.byType(DropdownButton2)); + await widgetTester.pumpAndSettle(); + + expect(find.text('One'), findsOneWidget); + expect(find.text('Two'), findsNWidgets(2)); + expect(find.text('Three'), findsOneWidget); + + await widgetTester.tap(find.text('One')); + await widgetTester.pumpAndSettle(); + + expect(find.text('One'), findsOneWidget); + expect(find.text('Two'), findsNothing); + expect(find.text('Three'), findsNothing); + + expect(comboBoxChooserModel.selectedChoice, 'One'); + expect(find.byIcon(Icons.priority_high), findsOneWidget); + + ntConnection.updateDataFromTopicName( + comboBoxChooserModel.activeTopicName, 'One'); + + comboBoxChooserModel.refresh(); + await widgetTester.pumpAndSettle(); + + expect(find.byIcon(Icons.priority_high), findsNothing); + expect(find.byIcon(Icons.check), findsOneWidget); + }); +} diff --git a/test/widgets/nt_widgets/multi-topic/command_scheduler_widget_test.dart b/test/widgets/nt_widgets/multi-topic/command_scheduler_widget_test.dart new file mode 100644 index 00000000..27030bb6 --- /dev/null +++ b/test/widgets/nt_widgets/multi-topic/command_scheduler_widget_test.dart @@ -0,0 +1,131 @@ +import 'package:flutter/material.dart'; + +import 'package:flutter_test/flutter_test.dart'; +import 'package:provider/provider.dart'; +import 'package:shared_preferences/shared_preferences.dart'; + +import 'package:elastic_dashboard/services/nt4_client.dart'; +import 'package:elastic_dashboard/services/nt_connection.dart'; +import 'package:elastic_dashboard/services/nt_widget_builder.dart'; +import 'package:elastic_dashboard/widgets/nt_widgets/multi-topic/command_scheduler.dart'; +import 'package:elastic_dashboard/widgets/nt_widgets/nt_widget.dart'; +import '../../../test_util.dart'; + +void main() { + TestWidgetsFlutterBinding.ensureInitialized(); + + final Map commandSchedulerJson = { + 'topic': 'Test/Command Scheduler', + 'period': 0.100, + }; + + late SharedPreferences preferences; + late NTConnection ntConnection; + + setUp(() async { + SharedPreferences.setMockInitialValues({}); + preferences = await SharedPreferences.getInstance(); + + ntConnection = createMockOnlineNT4( + virtualTopics: [ + NT4Topic( + name: 'Test/Command Scheduler/Names', + type: NT4TypeStr.kStringArr, + properties: {}), + NT4Topic( + name: 'Test/Command Scheduler/Ids', + type: NT4TypeStr.kIntArr, + properties: {}), + NT4Topic( + name: 'Test/Command Scheduler/Cancel', + type: NT4TypeStr.kIntArr, + properties: {}), + ], + virtualValues: { + 'Test/Command Scheduler/Names': ['Command 1', 'Command 2'], + 'Test/Command Scheduler/Ids': [1, 2], + }, + ); + }); + + test('Command scheduler from json', () { + NTWidgetModel commandSchedulerModel = NTWidgetBuilder.buildNTModelFromJson( + ntConnection, + preferences, + 'Scheduler', + commandSchedulerJson, + ); + + expect(commandSchedulerModel.type, 'Scheduler'); + expect(commandSchedulerModel.runtimeType, CommandSchedulerModel); + }); + + test('Command scheduler to json', () { + CommandSchedulerModel commandSchedulerModel = CommandSchedulerModel( + ntConnection: ntConnection, + preferences: preferences, + topic: 'Test/Command Scheduler', + period: 0.100, + ); + + expect(commandSchedulerModel.toJson(), commandSchedulerJson); + }); + + testWidgets('Command scheduler widget', (widgetTester) async { + FlutterError.onError = ignoreOverflowErrors; + + NTWidgetModel commandSchedulerModel = NTWidgetBuilder.buildNTModelFromJson( + ntConnection, + preferences, + 'Scheduler', + commandSchedulerJson, + ); + + await widgetTester.pumpWidget( + MaterialApp( + home: Scaffold( + body: ChangeNotifierProvider.value( + value: commandSchedulerModel, + child: const CommandSchedulerWidget(), + ), + ), + ), + ); + + await widgetTester.pumpAndSettle(); + + expect(find.byType(ListView), findsOneWidget); + expect(find.byType(ListTile), findsNWidgets(2)); + + expect(find.text('Command 1'), findsOneWidget); + expect(find.text('Command 2'), findsOneWidget); + + expect(find.text('ID: 1'), findsOneWidget); + expect(find.text('ID: 2'), findsOneWidget); + + expect(find.byIcon(Icons.cancel_outlined), findsNWidgets(2)); + + await widgetTester.tap(find.byIcon(Icons.cancel_outlined).first); + await widgetTester.pumpAndSettle(); + + expect(ntConnection.getLastAnnouncedValue('Test/Command Scheduler/Cancel'), + [1]); + + ntConnection.updateDataFromTopicName('Test/Command Scheduler/Ids', [2]); + ntConnection + .updateDataFromTopicName('Test/Command Scheduler/Names', ['Command 2']); + + commandSchedulerModel.refresh(); + await widgetTester.pumpAndSettle(); + + expect(find.byType(ListTile), findsOneWidget); + expect(find.text('Command 1'), findsNothing); + expect(find.text('ID: 1'), findsNothing); + + await widgetTester.tap(find.byIcon(Icons.cancel_outlined)); + await widgetTester.pumpAndSettle(); + + expect(ntConnection.getLastAnnouncedValue('Test/Command Scheduler/Cancel'), + [1, 2]); + }); +} diff --git a/test/widgets/nt_widgets/multi-topic/command_widget_test.dart b/test/widgets/nt_widgets/multi-topic/command_widget_test.dart new file mode 100644 index 00000000..87430a5b --- /dev/null +++ b/test/widgets/nt_widgets/multi-topic/command_widget_test.dart @@ -0,0 +1,124 @@ +import 'package:flutter/material.dart'; + +import 'package:flutter_test/flutter_test.dart'; +import 'package:provider/provider.dart'; +import 'package:shared_preferences/shared_preferences.dart'; + +import 'package:elastic_dashboard/services/nt4_client.dart'; +import 'package:elastic_dashboard/services/nt_connection.dart'; +import 'package:elastic_dashboard/services/nt_widget_builder.dart'; +import 'package:elastic_dashboard/widgets/nt_widgets/multi-topic/command_widget.dart'; +import 'package:elastic_dashboard/widgets/nt_widgets/nt_widget.dart'; +import '../../../test_util.dart'; + +void main() { + TestWidgetsFlutterBinding.ensureInitialized(); + + final Map commandWidgetJson = { + 'topic': 'Test/Command', + 'period': 0.100, + 'show_type': true, + }; + + late SharedPreferences preferences; + late NTConnection ntConnection; + + setUp(() async { + SharedPreferences.setMockInitialValues({}); + preferences = await SharedPreferences.getInstance(); + + ntConnection = createMockOnlineNT4( + virtualTopics: [ + NT4Topic( + name: 'Test/Command/running', + type: NT4TypeStr.kBool, + properties: {}, + ), + NT4Topic( + name: 'Test/Command/name', + type: NT4TypeStr.kString, + properties: {}, + ), + ], + virtualValues: { + 'Test/Commnad/running': false, + 'Test/Command/name': 'Test Command', + }, + ); + }); + + test('Command widget from json', () { + NTWidgetModel commandModel = NTWidgetBuilder.buildNTModelFromJson( + ntConnection, + preferences, + 'Command', + commandWidgetJson, + ); + + expect(commandModel.type, 'Command'); + expect(commandModel.runtimeType, CommandModel); + + if (commandModel is! CommandModel) { + return; + } + + expect(commandModel.showType, isTrue); + }); + + test('Command widget to json', () { + CommandModel commandModel = CommandModel( + ntConnection: ntConnection, + preferences: preferences, + topic: 'Test/Command', + period: 0.100, + showType: true, + ); + + expect(commandModel.toJson(), commandWidgetJson); + }); + + testWidgets('Command widget test', (widgetTester) async { + FlutterError.onError = ignoreOverflowErrors; + + NTWidgetModel commandModel = NTWidgetBuilder.buildNTModelFromJson( + ntConnection, + preferences, + 'Command', + commandWidgetJson, + ); + + await widgetTester.pumpWidget( + MaterialApp( + home: Scaffold( + body: ChangeNotifierProvider.value( + value: commandModel, + child: const CommandWidget(), + ), + ), + ), + ); + + await widgetTester.pumpAndSettle(); + + expect(find.text('Command'), findsOneWidget); + expect(find.text('Type: Test Command'), findsOneWidget); + + await widgetTester.tap(find.text('Command')); + await widgetTester.pumpAndSettle(); + + expect(ntConnection.getLastAnnouncedValue('Test/Command/running'), isTrue); + + commandModel.refresh(); + await widgetTester.pumpAndSettle(); + + await widgetTester.tap(find.text('Command')); + await widgetTester.pumpAndSettle(); + + expect(ntConnection.getLastAnnouncedValue('Test/Command/running'), isFalse); + + (commandModel as CommandModel).showType = false; + await widgetTester.pumpAndSettle(); + + expect(find.text('Type: Test Command'), findsNothing); + }); +} diff --git a/test/widgets/nt_widgets/multi-topic/differential_drive_test.dart b/test/widgets/nt_widgets/multi-topic/differential_drive_test.dart new file mode 100644 index 00000000..38d3e524 --- /dev/null +++ b/test/widgets/nt_widgets/multi-topic/differential_drive_test.dart @@ -0,0 +1,122 @@ +import 'package:flutter/material.dart'; + +import 'package:flutter_test/flutter_test.dart'; +import 'package:provider/provider.dart'; +import 'package:shared_preferences/shared_preferences.dart'; +import 'package:syncfusion_flutter_gauges/gauges.dart'; + +import 'package:elastic_dashboard/services/nt4_client.dart'; +import 'package:elastic_dashboard/services/nt_connection.dart'; +import 'package:elastic_dashboard/services/nt_widget_builder.dart'; +import 'package:elastic_dashboard/widgets/nt_widgets/multi-topic/differential_drive.dart'; +import 'package:elastic_dashboard/widgets/nt_widgets/nt_widget.dart'; +import '../../../test_util.dart'; + +void main() { + TestWidgetsFlutterBinding.ensureInitialized(); + + final Map differentialDriveJson = { + 'topic': 'Test/Differential Drive', + 'period': 0.100, + }; + + late SharedPreferences preferences; + late NTConnection ntConnection; + + setUp(() async { + SharedPreferences.setMockInitialValues({}); + preferences = await SharedPreferences.getInstance(); + + ntConnection = createMockOnlineNT4( + virtualTopics: [ + NT4Topic( + name: 'Test/Differential Drive/Left Motor Speed', + type: NT4TypeStr.kFloat32, + properties: {}, + ), + NT4Topic( + name: 'Test/Differential Drive/Right Motor Speed', + type: NT4TypeStr.kFloat32, + properties: {}, + ), + ], + virtualValues: { + 'Test/Differential Drive/Left Motor Speed': 0.50, + 'Test/Differential Drive/Right Motor Speed': 0.50, + }, + ); + }); + + test('Differential drive from json', () { + NTWidgetModel differentialDriveModel = NTWidgetBuilder.buildNTModelFromJson( + ntConnection, + preferences, + 'DifferentialDrive', + differentialDriveJson, + ); + + expect(differentialDriveModel.type, 'DifferentialDrive'); + expect(differentialDriveModel.runtimeType, DifferentialDriveModel); + }); + + test('Differential drive to json', () { + DifferentialDriveModel differentialDriveModel = DifferentialDriveModel( + ntConnection: ntConnection, + preferences: preferences, + topic: 'Test/Differential Drive', + period: 0.100, + ); + + expect(differentialDriveModel.toJson(), differentialDriveJson); + }); + + testWidgets('Differential drive widget test', (widgetTester) async { + FlutterError.onError = ignoreOverflowErrors; + + NTWidgetModel differentialDriveModel = NTWidgetBuilder.buildNTModelFromJson( + ntConnection, + preferences, + 'DifferentialDrive', + differentialDriveJson, + ); + + await widgetTester.pumpWidget( + MaterialApp( + home: Scaffold( + body: ChangeNotifierProvider.value( + value: differentialDriveModel, + child: const DifferentialDrive(), + ), + ), + ), + ); + + await widgetTester.pumpAndSettle(); + + expect(find.byType(CustomPaint), findsWidgets); + expect(find.byType(SfLinearGauge), findsNWidgets(2)); + expect(find.byType(LinearShapePointer), findsNWidgets(2)); + + await widgetTester.drag( + find.byType(LinearShapePointer).first, const Offset(0.0, 200.0)); + await widgetTester.pumpAndSettle(); + + expect( + ntConnection + .getLastAnnouncedValue('Test/Differential Drive/Left Motor Speed'), + isNot(0.50)); + expect( + ntConnection + .getLastAnnouncedValue('Test/Differential Drive/Right Motor Speed'), + 0.50); + + await widgetTester.drag( + find.byType(LinearShapePointer).last, const Offset(0.0, 300.0)); + await widgetTester.pumpAndSettle(); + + expect( + ntConnection + .getLastAnnouncedValue('Test/Differential Drive/Right Motor Speed'), + isNot(0.50)); + }); +} diff --git a/test/widgets/nt_widgets/multi-topic/encoder_widget_test.dart b/test/widgets/nt_widgets/multi-topic/encoder_widget_test.dart new file mode 100644 index 00000000..742b228d --- /dev/null +++ b/test/widgets/nt_widgets/multi-topic/encoder_widget_test.dart @@ -0,0 +1,119 @@ +import 'package:flutter/material.dart'; + +import 'package:flutter_test/flutter_test.dart'; +import 'package:provider/provider.dart'; +import 'package:shared_preferences/shared_preferences.dart'; + +import 'package:elastic_dashboard/services/nt4_client.dart'; +import 'package:elastic_dashboard/services/nt_connection.dart'; +import 'package:elastic_dashboard/services/nt_widget_builder.dart'; +import 'package:elastic_dashboard/widgets/nt_widgets/multi-topic/encoder_widget.dart'; +import 'package:elastic_dashboard/widgets/nt_widgets/nt_widget.dart'; +import '../../../test_util.dart'; + +void main() { + TestWidgetsFlutterBinding.ensureInitialized(); + + final Map encoderWidgetJson = { + 'topic': 'Test/Encoder', + 'period': 0.100, + }; + + late SharedPreferences preferences; + late NTConnection ntConnection; + + setUp(() async { + SharedPreferences.setMockInitialValues({}); + preferences = await SharedPreferences.getInstance(); + + ntConnection = createMockOnlineNT4( + virtualTopics: [ + NT4Topic( + name: 'Test/Encoder/Distance', + type: NT4TypeStr.kFloat32, + properties: {}), + NT4Topic( + name: 'Test/Encoder/Speed', + type: NT4TypeStr.kFloat32, + properties: {}), + ], + virtualValues: { + 'Test/Encoder/Distance': 5.50, + 'Test/Encoder/Speed': -10.0, + }, + ); + }); + + test('Encoder from json', () { + NTWidgetModel encoderWidgetModel = NTWidgetBuilder.buildNTModelFromJson( + ntConnection, + preferences, + 'Encoder', + encoderWidgetJson, + ); + + expect(encoderWidgetModel.type, 'Encoder'); + expect(encoderWidgetModel.runtimeType, EncoderModel); + }); + + test('Encoder alias name', () { + NTWidgetModel encoderWidgetModel = NTWidgetBuilder.buildNTModelFromJson( + ntConnection, + preferences, + 'Quadrature Encoder', + encoderWidgetJson, + ); + + expect(encoderWidgetModel.type, 'Encoder'); + expect(encoderWidgetModel.runtimeType, EncoderModel); + }); + + test('Encoder to json', () { + EncoderModel encoderWidgetModel = EncoderModel( + ntConnection: ntConnection, + preferences: preferences, + topic: 'Test/Encoder', + period: 0.100, + ); + + expect(encoderWidgetModel.toJson(), encoderWidgetJson); + }); + + testWidgets('Encoder widget test', (widgetTester) async { + FlutterError.onError = ignoreOverflowErrors; + + NTWidgetModel encoderWidgetModel = NTWidgetBuilder.buildNTModelFromJson( + ntConnection, + preferences, + 'Encoder', + encoderWidgetJson, + ); + + await widgetTester.pumpWidget( + MaterialApp( + home: Scaffold( + body: ChangeNotifierProvider.value( + value: encoderWidgetModel, + child: const EncoderWidget(), + ), + ), + ), + ); + + await widgetTester.pumpAndSettle(); + + expect(find.text('Distance'), findsOneWidget); + expect(find.text('Speed'), findsOneWidget); + + expect( + find.descendant( + of: find.byType(SelectableText), + matching: find.textContaining('5.50')), + findsOneWidget); + expect( + find.descendant( + of: find.byType(SelectableText), + matching: find.textContaining('-10.00')), + findsOneWidget); + }); +} diff --git a/test/widgets/nt_widgets/multi-topic/fms_info_test.dart b/test/widgets/nt_widgets/multi-topic/fms_info_test.dart new file mode 100644 index 00000000..d664cfe7 --- /dev/null +++ b/test/widgets/nt_widgets/multi-topic/fms_info_test.dart @@ -0,0 +1,309 @@ +import 'package:flutter/material.dart'; + +import 'package:flutter_test/flutter_test.dart'; +import 'package:provider/provider.dart'; +import 'package:shared_preferences/shared_preferences.dart'; + +import 'package:elastic_dashboard/services/nt4_client.dart'; +import 'package:elastic_dashboard/services/nt_connection.dart'; +import 'package:elastic_dashboard/services/nt_widget_builder.dart'; +import 'package:elastic_dashboard/widgets/nt_widgets/multi-topic/fms_info.dart'; +import 'package:elastic_dashboard/widgets/nt_widgets/nt_widget.dart'; +import '../../../test_util.dart'; + +void main() { + TestWidgetsFlutterBinding.ensureInitialized(); + + final Map fmsInfoJson = { + 'topic': 'Test/FMSInfo', + 'period': 0.100, + }; + + late SharedPreferences preferences; + late NTConnection ntConnection; + + NTConnection createNTConnection({ + String eventName = '', + bool redAlliance = false, + int matchNumber = 0, + int matchType = 0, + int replayNumber = 0, + bool enabled = false, + bool auto = false, + bool test = false, + bool estop = false, + bool fmsAttached = false, + bool dsAttached = false, + }) { + int fmsControlData = 0; + if (enabled) { + fmsControlData |= FMSInfo.ENABLED_FLAG; + } + if (auto) { + fmsControlData |= FMSInfo.AUTO_FLAG; + } + if (test) { + fmsControlData |= FMSInfo.TEST_FLAG; + } + if (estop) { + fmsControlData |= FMSInfo.EMERGENCY_STOP_FLAG; + } + if (fmsAttached) { + fmsControlData |= FMSInfo.FMS_ATTACHED_FLAG; + } + if (dsAttached) { + fmsControlData |= FMSInfo.DS_ATTACHED_FLAG; + } + + return createMockOnlineNT4( + virtualTopics: [ + NT4Topic( + name: 'Test/FMSInfo/EventName', + type: NT4TypeStr.kString, + properties: {}, + ), + NT4Topic( + name: 'Test/FMSInfo/FMSControlData', + type: NT4TypeStr.kInt, + properties: {}, + ), + NT4Topic( + name: 'Test/FMSInfo/IsRedAlliance', + type: NT4TypeStr.kBool, + properties: {}, + ), + NT4Topic( + name: 'Test/FMSInfo/MatchNumber', + type: NT4TypeStr.kInt, + properties: {}, + ), + NT4Topic( + name: 'Test/FMSInfo/MatchType', + type: NT4TypeStr.kInt, + properties: {}, + ), + NT4Topic( + name: 'Test/FMSInfo/ReplayNumber', + type: NT4TypeStr.kInt, + properties: {}, + ), + ], + virtualValues: { + 'Test/FMSInfo/EventName': eventName, + 'Test/FMSInfo/FMSControlData': fmsControlData, + 'Test/FMSInfo/IsRedAlliance': redAlliance, + 'Test/FMSInfo/MatchNumber': matchNumber, + 'Test/FMSInfo/MatchType': matchType, + 'Test/FMSInfo/ReplayNumber': replayNumber, + }, + ); + } + + Future pushFMSInfoWidget( + WidgetTester widgetTester, NTConnection ntConnection) async { + NTWidgetModel fmsInfoModel = NTWidgetBuilder.buildNTModelFromJson( + ntConnection, preferences, 'FMSInfo', fmsInfoJson); + + return widgetTester.pumpWidget( + MaterialApp( + home: Scaffold( + body: ChangeNotifierProvider.value( + value: fmsInfoModel, + child: const FMSInfo(), + ), + ), + ), + ); + } + + setUp(() async { + SharedPreferences.setMockInitialValues({}); + preferences = await SharedPreferences.getInstance(); + + ntConnection = createNTConnection( + eventName: 'CMPTX', + redAlliance: false, + matchNumber: 15, + matchType: 3, + replayNumber: 1, + enabled: true, + fmsAttached: true, + dsAttached: true, + ); + }); + + test('FMSInfo from json', () { + NTWidgetModel fmsInfoModel = NTWidgetBuilder.buildNTModelFromJson( + ntConnection, + preferences, + 'FMSInfo', + fmsInfoJson, + ); + + expect(fmsInfoModel.type, 'FMSInfo'); + expect(fmsInfoModel.runtimeType, FMSInfoModel); + }); + + test('FMSInfo to json', () { + FMSInfoModel fmsInfoModel = FMSInfoModel( + ntConnection: ntConnection, + preferences: preferences, + topic: 'Test/FMSInfo', + period: 0.100, + ); + + expect(fmsInfoModel.toJson(), fmsInfoJson); + }); + + testWidgets('FMSInfo CMPTX E15, Teleop Enabled', (widgetTester) async { + FlutterError.onError = ignoreOverflowErrors; + + await pushFMSInfoWidget(widgetTester, ntConnection); + + await widgetTester.pumpAndSettle(); + + expect(find.text('CMPTX Elimination match 15 (replay 1)'), findsOneWidget); + expect(find.text('DriverStation Connected'), findsOneWidget); + expect(find.text('FMS Connected'), findsOneWidget); + expect(find.byIcon(Icons.check), findsNWidgets(2)); + + expect(find.text('Robot State: Teleoperated'), findsOneWidget); + expect( + find.byWidgetPredicate((widget) => + widget is Container && + widget.decoration is BoxDecoration && + (widget.decoration as BoxDecoration).color == Colors.blue.shade900), + findsOneWidget); + }); + + testWidgets('FMSInfo NYSU Q72, Auto Enabled', (widgetTester) async { + FlutterError.onError = ignoreOverflowErrors; + + await pushFMSInfoWidget( + widgetTester, + createNTConnection( + eventName: 'NYSU', + redAlliance: false, + matchNumber: 72, + matchType: 2, + replayNumber: 1, + enabled: true, + auto: true, + fmsAttached: true, + dsAttached: true, + )); + + await widgetTester.pumpAndSettle(); + + expect(find.text('NYSU Qualification match 72 (replay 1)'), findsOneWidget); + expect(find.text('DriverStation Connected'), findsOneWidget); + expect(find.text('FMS Connected'), findsOneWidget); + expect(find.byIcon(Icons.check), findsNWidgets(2)); + + expect(find.text('Robot State: Autonomous'), findsOneWidget); + expect( + find.byWidgetPredicate((widget) => + widget is Container && + widget.decoration is BoxDecoration && + (widget.decoration as BoxDecoration).color == Colors.blue.shade900), + findsOneWidget); + }); + + testWidgets('FMSInfo NYLI2 P7, Estopped', (widgetTester) async { + FlutterError.onError = ignoreOverflowErrors; + + await pushFMSInfoWidget( + widgetTester, + createNTConnection( + eventName: 'NYLI2', + redAlliance: true, + matchNumber: 7, + matchType: 1, + replayNumber: 1, + estop: true, + fmsAttached: true, + dsAttached: true, + )); + + await widgetTester.pumpAndSettle(); + + expect(find.text('NYLI2 Practice match 7 (replay 1)'), findsOneWidget); + expect(find.text('DriverStation Connected'), findsOneWidget); + expect(find.text('FMS Connected'), findsOneWidget); + expect(find.byIcon(Icons.check), findsNWidgets(2)); + + expect(find.text('EMERGENCY STOPPED'), findsOneWidget); + expect(find.byType(CustomPaint), findsAtLeastNWidgets(2)); + expect( + find.byWidgetPredicate((widget) => + widget is Container && + widget.decoration is BoxDecoration && + (widget.decoration as BoxDecoration).color == Colors.red.shade900), + findsOneWidget); + }); + + testWidgets('FMSInfo Unkown Match, test enabled', (widgetTester) async { + FlutterError.onError = ignoreOverflowErrors; + + await pushFMSInfoWidget( + widgetTester, + createNTConnection( + eventName: '', + redAlliance: true, + matchNumber: 0, + matchType: 0, + replayNumber: 0, + enabled: true, + test: true, + fmsAttached: false, + dsAttached: true, + )); + + await widgetTester.pumpAndSettle(); + + expect(find.text('Unknown match 0'), findsOneWidget); + expect(find.text('DriverStation Connected'), findsOneWidget); + expect(find.text('FMS Disconnected'), findsOneWidget); + expect(find.byIcon(Icons.check), findsNWidgets(1)); + expect(find.byIcon(Icons.clear), findsNWidgets(1)); + + expect(find.text('Robot State: Test'), findsOneWidget); + expect( + find.byWidgetPredicate((widget) => + widget is Container && + widget.decoration is BoxDecoration && + (widget.decoration as BoxDecoration).color == Colors.red.shade900), + findsOneWidget); + }); + + testWidgets('FMSInfo Unknown Match, everything disconnected', + (widgetTester) async { + FlutterError.onError = ignoreOverflowErrors; + + await pushFMSInfoWidget( + widgetTester, + createNTConnection( + eventName: '', + redAlliance: true, + matchNumber: 0, + matchType: 0, + replayNumber: 0, + )); + + await widgetTester.pumpAndSettle(); + + expect(find.text('Unknown match 0'), findsOneWidget); + expect(find.text('DriverStation Disconnected'), findsOneWidget); + expect(find.text('FMS Disconnected'), findsOneWidget); + expect(find.byIcon(Icons.check), findsNWidgets(0)); + expect(find.byIcon(Icons.clear), findsNWidgets(2)); + + expect(find.text('Robot State: Disabled'), findsOneWidget); + expect( + find.byWidgetPredicate((widget) => + widget is Container && + widget.decoration is BoxDecoration && + (widget.decoration as BoxDecoration).color == Colors.red.shade900), + findsOneWidget); + }); +} diff --git a/test/widgets/nt_widgets/multi-topic/gyro_test.dart b/test/widgets/nt_widgets/multi-topic/gyro_test.dart new file mode 100644 index 00000000..e110daf5 --- /dev/null +++ b/test/widgets/nt_widgets/multi-topic/gyro_test.dart @@ -0,0 +1,101 @@ +import 'package:flutter/material.dart'; + +import 'package:flutter_test/flutter_test.dart'; +import 'package:provider/provider.dart'; +import 'package:shared_preferences/shared_preferences.dart'; +import 'package:syncfusion_flutter_gauges/gauges.dart'; + +import 'package:elastic_dashboard/services/nt4_client.dart'; +import 'package:elastic_dashboard/services/nt_connection.dart'; +import 'package:elastic_dashboard/services/nt_widget_builder.dart'; +import 'package:elastic_dashboard/widgets/nt_widgets/multi-topic/gyro.dart'; +import 'package:elastic_dashboard/widgets/nt_widgets/nt_widget.dart'; +import '../../../test_util.dart'; + +void main() { + TestWidgetsFlutterBinding.ensureInitialized(); + + final Map gyroJson = { + 'topic': 'Test/Gyro', + 'period': 0.100, + 'counter_clockwise_positive': true, + }; + + late SharedPreferences preferences; + late NTConnection ntConnection; + + setUp(() async { + SharedPreferences.setMockInitialValues({}); + preferences = await SharedPreferences.getInstance(); + + ntConnection = createMockOnlineNT4( + virtualTopics: [ + NT4Topic( + name: 'Test/Gyro/Value', + type: NT4TypeStr.kFloat32, + properties: {}, + ), + ], + virtualValues: { + 'Test/Gyro/Value': 183.5, + }, + ); + }); + + test('Gyro from json', () { + NTWidgetModel gyroModel = NTWidgetBuilder.buildNTModelFromJson( + ntConnection, + preferences, + 'Gyro', + gyroJson, + ); + + expect(gyroModel.type, 'Gyro'); + expect(gyroModel.runtimeType, GyroModel); + + if (gyroModel is! GyroModel) { + return; + } + + expect(gyroModel.counterClockwisePositive, isTrue); + }); + + test('Gyro to json', () { + GyroModel gyroModel = GyroModel( + ntConnection: ntConnection, + preferences: preferences, + topic: 'Test/Gyro', + period: 0.100, + counterClockwisePositive: true, + ); + + expect(gyroModel.toJson(), gyroJson); + }); + + testWidgets('Gyro widget', (widgetTester) async { + FlutterError.onError = ignoreOverflowErrors; + + NTWidgetModel gyroModel = NTWidgetBuilder.buildNTModelFromJson( + ntConnection, + preferences, + 'Gyro', + gyroJson, + ); + + await widgetTester.pumpWidget( + MaterialApp( + home: Scaffold( + body: ChangeNotifierProvider.value( + value: gyroModel, + child: const Gyro(), + ), + ), + ), + ); + + await widgetTester.pumpAndSettle(); + + expect(find.text('176.50'), findsOneWidget); + expect(find.byType(SfRadialGauge), findsOneWidget); + }); +} diff --git a/test/widgets/nt_widgets/multi-topic/motor_controller_test.dart b/test/widgets/nt_widgets/multi-topic/motor_controller_test.dart new file mode 100644 index 00000000..306d9b10 --- /dev/null +++ b/test/widgets/nt_widgets/multi-topic/motor_controller_test.dart @@ -0,0 +1,106 @@ +import 'package:flutter/material.dart'; + +import 'package:flutter_test/flutter_test.dart'; +import 'package:provider/provider.dart'; +import 'package:shared_preferences/shared_preferences.dart'; +import 'package:syncfusion_flutter_gauges/gauges.dart'; + +import 'package:elastic_dashboard/services/nt4_client.dart'; +import 'package:elastic_dashboard/services/nt_connection.dart'; +import 'package:elastic_dashboard/services/nt_widget_builder.dart'; +import 'package:elastic_dashboard/widgets/nt_widgets/multi-topic/motor_controller.dart'; +import 'package:elastic_dashboard/widgets/nt_widgets/nt_widget.dart'; +import '../../../test_util.dart'; + +void main() { + TestWidgetsFlutterBinding.ensureInitialized(); + + final Map motorControllerJson = { + 'topic': 'Test/Motor Controller', + 'period': 0.100, + }; + + late SharedPreferences preferences; + late NTConnection ntConnection; + + setUp(() async { + SharedPreferences.setMockInitialValues({}); + preferences = await SharedPreferences.getInstance(); + + ntConnection = createMockOnlineNT4( + virtualTopics: [ + NT4Topic( + name: 'Test/Motor Controller/Value', + type: NT4TypeStr.kFloat32, + properties: {}, + ), + ], + virtualValues: { + 'Test/Motor Controller/Value': -0.5, + }, + ); + }); + + test('Motor controller from json', () { + NTWidgetModel motorControllerModel = NTWidgetBuilder.buildNTModelFromJson( + ntConnection, + preferences, + 'Motor Controller', + motorControllerJson, + ); + + expect(motorControllerModel.type, 'Motor Controller'); + expect(motorControllerModel.runtimeType, MotorControllerModel); + }); + + test('Nidec brushless from json', () { + NTWidgetModel motorControllerModel = NTWidgetBuilder.buildNTModelFromJson( + ntConnection, + preferences, + 'Nidec Brushless', + motorControllerJson, + ); + + expect(motorControllerModel.type, 'Motor Controller'); + expect(motorControllerModel.runtimeType, MotorControllerModel); + }); + + test('Motor controller to json', () { + MotorControllerModel motorControllerModel = MotorControllerModel( + ntConnection: ntConnection, + preferences: preferences, + topic: 'Test/Motor Controller', + period: 0.100, + ); + + expect(motorControllerModel.toJson(), motorControllerJson); + }); + + testWidgets('Motor controller widget', (widgetTester) async { + FlutterError.onError = ignoreOverflowErrors; + + NTWidgetModel motorControllerModel = NTWidgetBuilder.buildNTModelFromJson( + ntConnection, + preferences, + 'Motor Controller', + motorControllerJson, + ); + + await widgetTester.pumpWidget( + MaterialApp( + home: Scaffold( + body: ChangeNotifierProvider.value( + value: motorControllerModel, + child: const MotorController(), + ), + ), + ), + ); + + await widgetTester.pumpAndSettle(); + + expect(find.text('-0.50'), findsOneWidget); + expect(find.byType(SfLinearGauge), findsOneWidget); + expect(find.byType(LinearShapePointer), findsOneWidget); + }); +} diff --git a/test/widgets/nt_widgets/multi-topic/network_alerts_test.dart b/test/widgets/nt_widgets/multi-topic/network_alerts_test.dart new file mode 100644 index 00000000..7263ba11 --- /dev/null +++ b/test/widgets/nt_widgets/multi-topic/network_alerts_test.dart @@ -0,0 +1,114 @@ +import 'package:flutter/material.dart'; + +import 'package:flutter_test/flutter_test.dart'; +import 'package:provider/provider.dart'; +import 'package:shared_preferences/shared_preferences.dart'; + +import 'package:elastic_dashboard/services/nt4_client.dart'; +import 'package:elastic_dashboard/services/nt_connection.dart'; +import 'package:elastic_dashboard/services/nt_widget_builder.dart'; +import 'package:elastic_dashboard/widgets/nt_widgets/multi-topic/network_alerts.dart'; +import 'package:elastic_dashboard/widgets/nt_widgets/nt_widget.dart'; +import '../../../test_util.dart'; + +void main() { + TestWidgetsFlutterBinding.ensureInitialized(); + + final Map networkAlertsJson = { + 'topic': 'Test/Alerts', + 'period': 0.100, + }; + + late SharedPreferences preferences; + late NTConnection ntConnection; + + setUp(() async { + SharedPreferences.setMockInitialValues({}); + preferences = await SharedPreferences.getInstance(); + + ntConnection = createMockOnlineNT4( + virtualTopics: [ + NT4Topic( + name: 'Test/Alerts/errors', + type: NT4TypeStr.kStringArr, + properties: {}, + ), + NT4Topic( + name: 'Test/Alerts/warnings', + type: NT4TypeStr.kStringArr, + properties: {}, + ), + NT4Topic( + name: 'Test/Alerts/infos', + type: NT4TypeStr.kStringArr, + properties: {}, + ), + ], + virtualValues: { + 'Test/Alerts/errors': ['Test Error 1', 'Test Error 2'], + 'Test/Alerts/warnings': ['Test Warning 1', 'Test Warning 2'], + 'Test/Alerts/infos': ['Test Info 1', 'Test Info 2'], + }, + ); + }); + + test('Network alerts from json', () { + NTWidgetModel networkAlertsModel = NTWidgetBuilder.buildNTModelFromJson( + ntConnection, + preferences, + 'Alerts', + networkAlertsJson, + ); + + expect(networkAlertsModel.type, 'Alerts'); + expect(networkAlertsModel.runtimeType, NetworkAlertsModel); + }); + + test('Network alerts to json', () { + NetworkAlertsModel networkAlertsModel = NetworkAlertsModel( + ntConnection: ntConnection, + preferences: preferences, + topic: 'Test/Alerts', + period: 0.100, + ); + + expect(networkAlertsModel.toJson(), networkAlertsJson); + }); + + testWidgets('Network alerts widget', (widgetTester) async { + FlutterError.onError = ignoreOverflowErrors; + + NTWidgetModel networkAlertsModel = NTWidgetBuilder.buildNTModelFromJson( + ntConnection, + preferences, + 'Alerts', + networkAlertsJson, + ); + + await widgetTester.pumpWidget( + MaterialApp( + home: Scaffold( + body: ChangeNotifierProvider.value( + value: networkAlertsModel, + child: const NetworkAlerts(), + ), + ), + ), + ); + + await widgetTester.pumpAndSettle(); + + expect(find.byIcon(Icons.cancel), findsNWidgets(2)); + expect(find.byIcon(Icons.warning), findsNWidgets(2)); + expect(find.byIcon(Icons.info), findsNWidgets(2)); + + expect(find.text('Test Error 1'), findsOneWidget); + expect(find.text('Test Error 2'), findsOneWidget); + + expect(find.text('Test Warning 1'), findsOneWidget); + expect(find.text('Test Warning 2'), findsOneWidget); + + expect(find.text('Test Info 1'), findsOneWidget); + expect(find.text('Test Info 2'), findsOneWidget); + }); +} diff --git a/test/widgets/nt_widgets/multi-topic/pid_controller_test.dart b/test/widgets/nt_widgets/multi-topic/pid_controller_test.dart new file mode 100644 index 00000000..7a7af810 --- /dev/null +++ b/test/widgets/nt_widgets/multi-topic/pid_controller_test.dart @@ -0,0 +1,168 @@ +import 'package:flutter/material.dart'; + +import 'package:flutter_test/flutter_test.dart'; +import 'package:provider/provider.dart'; +import 'package:shared_preferences/shared_preferences.dart'; + +import 'package:elastic_dashboard/services/nt4_client.dart'; +import 'package:elastic_dashboard/services/nt_connection.dart'; +import 'package:elastic_dashboard/services/nt_widget_builder.dart'; +import 'package:elastic_dashboard/widgets/nt_widgets/multi-topic/pid_controller.dart'; +import 'package:elastic_dashboard/widgets/nt_widgets/nt_widget.dart'; +import '../../../test_util.dart'; + +void main() { + TestWidgetsFlutterBinding.ensureInitialized(); + + final Map pidControllerJson = { + 'topic': 'Test/PID Controller', + 'period': 0.100, + }; + + late SharedPreferences preferences; + late NTConnection ntConnection; + + setUp(() async { + SharedPreferences.setMockInitialValues({}); + preferences = await SharedPreferences.getInstance(); + + ntConnection = createMockOnlineNT4( + virtualTopics: [ + NT4Topic( + name: 'Test/PID Controller/p', + type: NT4TypeStr.kFloat32, + properties: {}, + ), + NT4Topic( + name: 'Test/PID Controller/i', + type: NT4TypeStr.kFloat32, + properties: {}, + ), + NT4Topic( + name: 'Test/PID Controller/d', + type: NT4TypeStr.kFloat32, + properties: {}, + ), + NT4Topic( + name: 'Test/PID Controller/setpoint', + type: NT4TypeStr.kFloat32, + properties: {}, + ), + ], + virtualValues: { + 'Test/PID Controller/p': 0.0, + 'Test/PID Controller/i': 0.0, + 'Test/PID Controller/d': 0.0, + 'Test/PID Controller/setpoint': 0.0, + }, + ); + }); + + test('PID controller from json', () { + NTWidgetModel pidControllerModel = NTWidgetBuilder.buildNTModelFromJson( + ntConnection, + preferences, + 'PIDController', + pidControllerJson, + ); + + expect(pidControllerModel.type, 'PIDController'); + expect(pidControllerModel.runtimeType, PIDControllerModel); + }); + + test('PID controller from alias name', () { + NTWidgetModel pidControllerModel = NTWidgetBuilder.buildNTModelFromJson( + ntConnection, + preferences, + 'PID Controller', + pidControllerJson, + ); + + expect(pidControllerModel.type, 'PIDController'); + expect(pidControllerModel.runtimeType, PIDControllerModel); + }); + + test('PID controller to json', () { + PIDControllerModel pidControllerModel = PIDControllerModel( + ntConnection: ntConnection, + preferences: preferences, + topic: 'Test/PID Controller', + period: 0.100, + ); + + expect(pidControllerModel.toJson(), pidControllerJson); + }); + + testWidgets('PID controller widget test', (widgetTester) async { + FlutterError.onError = ignoreOverflowErrors; + + NTWidgetModel pidControllerModel = NTWidgetBuilder.buildNTModelFromJson( + ntConnection, + preferences, + 'PIDController', + pidControllerJson, + ); + + await widgetTester.pumpWidget( + MaterialApp( + home: Scaffold( + body: ChangeNotifierProvider.value( + value: pidControllerModel, + child: const PIDControllerWidget(), + ), + ), + ), + ); + + await widgetTester.pumpAndSettle(); + + expect(find.byType(TextField), findsNWidgets(4)); + expect(find.widgetWithText(TextField, 'kP'), findsOneWidget); + expect(find.widgetWithText(TextField, 'kI'), findsOneWidget); + expect(find.widgetWithText(TextField, 'kD'), findsOneWidget); + expect(find.widgetWithText(TextField, 'Setpoint'), findsOneWidget); + + await widgetTester.enterText(find.widgetWithText(TextField, 'kP'), '0.100'); + await widgetTester.testTextInput.receiveAction(TextInputAction.done); + + expect(ntConnection.getLastAnnouncedValue('Test/PID Controller/p'), 0.0); + + await widgetTester.enterText(find.widgetWithText(TextField, 'kI'), '0.100'); + await widgetTester.testTextInput.receiveAction(TextInputAction.done); + + expect(ntConnection.getLastAnnouncedValue('Test/PID Controller/i'), 0.0); + + await widgetTester.enterText(find.widgetWithText(TextField, 'kD'), '0.100'); + await widgetTester.testTextInput.receiveAction(TextInputAction.done); + + expect(ntConnection.getLastAnnouncedValue('Test/PID Controller/d'), 0.0); + + await widgetTester.enterText( + find.widgetWithText(TextField, 'Setpoint'), '0.100'); + await widgetTester.testTextInput.receiveAction(TextInputAction.done); + + expect(ntConnection.getLastAnnouncedValue('Test/PID Controller/setpoint'), + 0.0); + + pidControllerModel.refresh(); + await widgetTester.pumpAndSettle(); + + expect(find.byIcon(Icons.priority_high), findsOneWidget); + + expect( + find.widgetWithText(OutlinedButton, 'Publish Values'), findsOneWidget); + await widgetTester + .tap(find.widgetWithText(OutlinedButton, 'Publish Values')); + + pidControllerModel.refresh(); + await widgetTester.pumpAndSettle(); + + expect(ntConnection.getLastAnnouncedValue('Test/PID Controller/p'), 0.1); + expect(ntConnection.getLastAnnouncedValue('Test/PID Controller/i'), 0.1); + expect(ntConnection.getLastAnnouncedValue('Test/PID Controller/d'), 0.1); + expect(ntConnection.getLastAnnouncedValue('Test/PID Controller/setpoint'), + 0.1); + + expect(find.byIcon(Icons.priority_high), findsNothing); + }); +} diff --git a/test/widgets/nt_widgets/multi-topic/power_distribution_test.dart b/test/widgets/nt_widgets/multi-topic/power_distribution_test.dart new file mode 100644 index 00000000..f8ace9d6 --- /dev/null +++ b/test/widgets/nt_widgets/multi-topic/power_distribution_test.dart @@ -0,0 +1,138 @@ +import 'package:flutter/material.dart'; + +import 'package:flutter_test/flutter_test.dart'; +import 'package:provider/provider.dart'; +import 'package:shared_preferences/shared_preferences.dart'; + +import 'package:elastic_dashboard/services/nt4_client.dart'; +import 'package:elastic_dashboard/services/nt_connection.dart'; +import 'package:elastic_dashboard/services/nt_widget_builder.dart'; +import 'package:elastic_dashboard/widgets/nt_widgets/multi-topic/power_distribution.dart'; +import 'package:elastic_dashboard/widgets/nt_widgets/nt_widget.dart'; +import '../../../test_util.dart'; + +void main() { + TestWidgetsFlutterBinding.ensureInitialized(); + + final Map powerDistributionJson = { + 'topic': 'Test/Power Distribution', + 'period': 0.100, + }; + + late SharedPreferences preferences; + late NTConnection ntConnection; + + setUp(() async { + SharedPreferences.setMockInitialValues({}); + preferences = await SharedPreferences.getInstance(); + + List channelTopics = []; + Map virtualChannelValues = {}; + + for (int i = 0; i <= PowerDistributionModel.numberOfChannels; i++) { + channelTopics.add(NT4Topic( + name: 'Test/Power Distribution/Chan$i', + type: NT4TypeStr.kFloat32, + properties: {}, + )); + + virtualChannelValues.addAll({'Test/Power Distribution/Chan$i': 0.00}); + } + + ntConnection = createMockOnlineNT4( + virtualTopics: [ + NT4Topic( + name: 'Test/Power Distribution/Voltage', + type: NT4TypeStr.kFloat32, + properties: {}, + ), + NT4Topic( + name: 'Test/Power Distribution/TotalCurrent', + type: NT4TypeStr.kFloat32, + properties: {}, + ), + ...channelTopics, + ], + virtualValues: { + 'Test/Power Distribution/Voltage': 12.00, + 'Test/Power Distribution/TotalCurrent': 100.0, + ...virtualChannelValues, + }, + ); + }); + + test('Power distribution from json', () { + NTWidgetModel powerDistributionModel = NTWidgetBuilder.buildNTModelFromJson( + ntConnection, + preferences, + 'PowerDistribution', + powerDistributionJson, + ); + + expect(powerDistributionModel.type, 'PowerDistribution'); + expect(powerDistributionModel.runtimeType, PowerDistributionModel); + }); + + test('Power distribution from alias name', () { + NTWidgetModel powerDistributionModel = NTWidgetBuilder.buildNTModelFromJson( + ntConnection, + preferences, + 'PDP', + powerDistributionJson, + ); + + expect(powerDistributionModel.type, 'PowerDistribution'); + expect(powerDistributionModel.runtimeType, PowerDistributionModel); + }); + + test('Power distribution to json', () { + PowerDistributionModel powerDistributionModel = PowerDistributionModel( + ntConnection: ntConnection, + preferences: preferences, + topic: 'Test/Power Distribution', + period: 0.100, + ); + + expect(powerDistributionModel.toJson(), powerDistributionJson); + }); + + testWidgets('Power distribution widget test', (widgetTester) async { + FlutterError.onError = ignoreOverflowErrors; + + NTWidgetModel powerDistributionModel = NTWidgetBuilder.buildNTModelFromJson( + ntConnection, + preferences, + 'PowerDistribution', + powerDistributionJson, + ); + + await widgetTester.pumpWidget( + MaterialApp( + home: Scaffold( + body: ChangeNotifierProvider.value( + value: powerDistributionModel, + child: const PowerDistribution(), + ), + ), + ), + ); + + await widgetTester.pumpAndSettle(); + + expect(find.text('Voltage'), findsOneWidget); + expect(find.text('12.00 V'), findsOneWidget); + + expect(find.text('Total Current'), findsOneWidget); + expect(find.text('100.00 A'), findsOneWidget); + + expect(find.text('00.00 A'), findsNWidgets(24)); + + for (int i = 0; i <= 9; i++) { + expect(find.text('Ch. $i '), findsOneWidget); + } + + for (int i = 10; i <= 23; i++) { + expect(find.text('Ch. $i'), findsOneWidget); + } + }); +} diff --git a/test/widgets/nt_widgets/multi-topic/profiled_pid_controller_test.dart b/test/widgets/nt_widgets/multi-topic/profiled_pid_controller_test.dart new file mode 100644 index 00000000..f9f3168d --- /dev/null +++ b/test/widgets/nt_widgets/multi-topic/profiled_pid_controller_test.dart @@ -0,0 +1,167 @@ +import 'package:flutter/material.dart'; + +import 'package:flutter_test/flutter_test.dart'; +import 'package:provider/provider.dart'; +import 'package:shared_preferences/shared_preferences.dart'; + +import 'package:elastic_dashboard/services/nt4_client.dart'; +import 'package:elastic_dashboard/services/nt_connection.dart'; +import 'package:elastic_dashboard/services/nt_widget_builder.dart'; +import 'package:elastic_dashboard/widgets/nt_widgets/multi-topic/profiled_pid_controller.dart'; +import 'package:elastic_dashboard/widgets/nt_widgets/nt_widget.dart'; +import '../../../test_util.dart'; + +void main() { + TestWidgetsFlutterBinding.ensureInitialized(); + + final Map profiledPIDControllerJson = { + 'topic': 'Test/Profiled PID Controller', + 'period': 0.100, + }; + + late SharedPreferences preferences; + late NTConnection ntConnection; + + setUp(() async { + SharedPreferences.setMockInitialValues({}); + preferences = await SharedPreferences.getInstance(); + + ntConnection = createMockOnlineNT4( + virtualTopics: [ + NT4Topic( + name: 'Test/Profiled PID Controller/p', + type: NT4TypeStr.kFloat32, + properties: {}, + ), + NT4Topic( + name: 'Test/Profiled PID Controller/i', + type: NT4TypeStr.kFloat32, + properties: {}, + ), + NT4Topic( + name: 'Test/Profiled PID Controller/d', + type: NT4TypeStr.kFloat32, + properties: {}, + ), + NT4Topic( + name: 'Test/Profiled PID Controller/goal', + type: NT4TypeStr.kFloat32, + properties: {}, + ), + ], + virtualValues: { + 'Test/Profiled PID Controller/p': 0.0, + 'Test/Profiled PID Controller/i': 0.0, + 'Test/Profiled PID Controller/d': 0.0, + 'Test/Profiled PID Controller/goal': 0.0, + }, + ); + }); + + test('PID controller from json', () { + NTWidgetModel profiledPIDControllerModel = + NTWidgetBuilder.buildNTModelFromJson( + ntConnection, + preferences, + 'ProfiledPIDController', + profiledPIDControllerJson, + ); + + expect(profiledPIDControllerModel.type, 'ProfiledPIDController'); + expect(profiledPIDControllerModel.runtimeType, ProfiledPIDControllerModel); + }); + + test('Profiled PID controller to json', () { + ProfiledPIDControllerModel profiledPIDControllerModel = + ProfiledPIDControllerModel( + ntConnection: ntConnection, + preferences: preferences, + topic: 'Test/Profiled PID Controller', + period: 0.100, + ); + + expect(profiledPIDControllerModel.toJson(), profiledPIDControllerJson); + }); + + testWidgets('Profiled PID controller widget test', (widgetTester) async { + FlutterError.onError = ignoreOverflowErrors; + + NTWidgetModel profiledPIDControllerModel = + NTWidgetBuilder.buildNTModelFromJson( + ntConnection, + preferences, + 'ProfiledPIDController', + profiledPIDControllerJson, + ); + + await widgetTester.pumpWidget( + MaterialApp( + home: Scaffold( + body: ChangeNotifierProvider.value( + value: profiledPIDControllerModel, + child: const ProfiledPIDControllerWidget(), + ), + ), + ), + ); + + await widgetTester.pumpAndSettle(); + + expect(find.byType(TextField), findsNWidgets(4)); + expect(find.widgetWithText(TextField, 'kP'), findsOneWidget); + expect(find.widgetWithText(TextField, 'kI'), findsOneWidget); + expect(find.widgetWithText(TextField, 'kD'), findsOneWidget); + expect(find.widgetWithText(TextField, 'Goal'), findsOneWidget); + + await widgetTester.enterText(find.widgetWithText(TextField, 'kP'), '0.100'); + await widgetTester.testTextInput.receiveAction(TextInputAction.done); + + expect(ntConnection.getLastAnnouncedValue('Test/Profiled PID Controller/p'), + 0.0); + + await widgetTester.enterText(find.widgetWithText(TextField, 'kI'), '0.100'); + await widgetTester.testTextInput.receiveAction(TextInputAction.done); + + expect(ntConnection.getLastAnnouncedValue('Test/Profiled PID Controller/i'), + 0.0); + + await widgetTester.enterText(find.widgetWithText(TextField, 'kD'), '0.100'); + await widgetTester.testTextInput.receiveAction(TextInputAction.done); + + expect(ntConnection.getLastAnnouncedValue('Test/Profiled PID Controller/d'), + 0.0); + + await widgetTester.enterText( + find.widgetWithText(TextField, 'Goal'), '0.100'); + await widgetTester.testTextInput.receiveAction(TextInputAction.done); + + expect( + ntConnection.getLastAnnouncedValue('Test/Profiled PID Controller/goal'), + 0.0); + + profiledPIDControllerModel.refresh(); + await widgetTester.pumpAndSettle(); + + expect(find.byIcon(Icons.priority_high), findsOneWidget); + + expect( + find.widgetWithText(OutlinedButton, 'Publish Values'), findsOneWidget); + await widgetTester + .tap(find.widgetWithText(OutlinedButton, 'Publish Values')); + + profiledPIDControllerModel.refresh(); + await widgetTester.pumpAndSettle(); + + expect(ntConnection.getLastAnnouncedValue('Test/Profiled PID Controller/p'), + 0.1); + expect(ntConnection.getLastAnnouncedValue('Test/Profiled PID Controller/i'), + 0.1); + expect(ntConnection.getLastAnnouncedValue('Test/Profiled PID Controller/d'), + 0.1); + expect( + ntConnection.getLastAnnouncedValue('Test/Profiled PID Controller/goal'), + 0.1); + + expect(find.byIcon(Icons.priority_high), findsNothing); + }); +} diff --git a/test/widgets/nt_widgets/multi-topic/relay_widget_test.dart b/test/widgets/nt_widgets/multi-topic/relay_widget_test.dart new file mode 100644 index 00000000..3fc070d2 --- /dev/null +++ b/test/widgets/nt_widgets/multi-topic/relay_widget_test.dart @@ -0,0 +1,115 @@ +import 'package:flutter/material.dart'; + +import 'package:flutter_test/flutter_test.dart'; +import 'package:provider/provider.dart'; +import 'package:shared_preferences/shared_preferences.dart'; + +import 'package:elastic_dashboard/services/nt4_client.dart'; +import 'package:elastic_dashboard/services/nt_connection.dart'; +import 'package:elastic_dashboard/services/nt_widget_builder.dart'; +import 'package:elastic_dashboard/widgets/nt_widgets/multi-topic/relay_widget.dart'; +import 'package:elastic_dashboard/widgets/nt_widgets/nt_widget.dart'; +import '../../../test_util.dart'; + +void main() { + TestWidgetsFlutterBinding.ensureInitialized(); + + final Map relayJson = { + 'topic': 'Test/Relay', + 'period': 0.100, + }; + + late SharedPreferences preferences; + late NTConnection ntConnection; + + setUp(() async { + SharedPreferences.setMockInitialValues({}); + preferences = await SharedPreferences.getInstance(); + + ntConnection = createMockOnlineNT4( + virtualTopics: [ + NT4Topic( + name: 'Test/Relay/Value', + type: NT4TypeStr.kString, + properties: {}, + ), + ], + virtualValues: { + 'Test/Relay/Value': 'Off', + }, + ); + }); + + test('Relay from json', () { + NTWidgetModel relayModel = NTWidgetBuilder.buildNTModelFromJson( + ntConnection, + preferences, + 'Relay', + relayJson, + ); + + expect(relayModel.type, 'Relay'); + expect(relayModel.runtimeType, RelayModel); + }); + + test('Relay to json', () { + RelayModel relayModel = RelayModel( + ntConnection: ntConnection, + preferences: preferences, + topic: 'Test/Relay', + period: 0.100, + ); + + expect(relayModel.toJson(), relayJson); + }); + + testWidgets('Relay widget test', (widgetTester) async { + FlutterError.onError = ignoreOverflowErrors; + + NTWidgetModel relayModel = NTWidgetBuilder.buildNTModelFromJson( + ntConnection, + preferences, + 'Relay', + relayJson, + ); + + await widgetTester.pumpWidget( + MaterialApp( + home: Scaffold( + body: ChangeNotifierProvider.value( + value: relayModel, + child: const RelayWidget(), + ), + ), + ), + ); + + await widgetTester.pumpAndSettle(); + + expect(find.byType(ToggleButtons), findsOneWidget); + + expect(find.text('On'), findsOneWidget); + await widgetTester.tap(find.text('On')); + await widgetTester.pumpAndSettle(); + + expect(ntConnection.getLastAnnouncedValue('Test/Relay/Value'), 'On'); + + expect(find.text('Forward'), findsOneWidget); + await widgetTester.tap(find.text('Forward')); + await widgetTester.pumpAndSettle(); + + expect(ntConnection.getLastAnnouncedValue('Test/Relay/Value'), 'Forward'); + + expect(find.text('Reverse'), findsOneWidget); + await widgetTester.tap(find.text('Reverse')); + await widgetTester.pumpAndSettle(); + + expect(ntConnection.getLastAnnouncedValue('Test/Relay/Value'), 'Reverse'); + + expect(find.text('Off'), findsOneWidget); + await widgetTester.tap(find.text('Off')); + await widgetTester.pumpAndSettle(); + + expect(ntConnection.getLastAnnouncedValue('Test/Relay/Value'), 'Off'); + }); +} diff --git a/test/widgets/nt_widgets/multi-topic/robot_preferences_test.dart b/test/widgets/nt_widgets/multi-topic/robot_preferences_test.dart new file mode 100644 index 00000000..9df07cc5 --- /dev/null +++ b/test/widgets/nt_widgets/multi-topic/robot_preferences_test.dart @@ -0,0 +1,158 @@ +import 'package:flutter/material.dart'; + +import 'package:flutter_test/flutter_test.dart'; +import 'package:provider/provider.dart'; +import 'package:shared_preferences/shared_preferences.dart'; + +import 'package:elastic_dashboard/services/nt4_client.dart'; +import 'package:elastic_dashboard/services/nt_connection.dart'; +import 'package:elastic_dashboard/services/nt_widget_builder.dart'; +import 'package:elastic_dashboard/widgets/nt_widgets/multi-topic/robot_preferences.dart'; +import 'package:elastic_dashboard/widgets/nt_widgets/nt_widget.dart'; +import '../../../test_util.dart'; + +void main() { + TestWidgetsFlutterBinding.ensureInitialized(); + + final Map robotPreferencesJson = { + 'topic': 'Test/Preferences', + 'period': 0.100, + }; + + late SharedPreferences preferences; + late NTConnection ntConnection; + + setUp(() async { + SharedPreferences.setMockInitialValues({}); + preferences = await SharedPreferences.getInstance(); + + ntConnection = createMockOnlineNT4( + virtualTopics: [ + NT4Topic( + name: 'Test/Preferences/Test Preference', + type: NT4TypeStr.kInt, + properties: {}, + ), + NT4Topic( + name: 'Test/Preferences/Preference 1', + type: NT4TypeStr.kFloat32, + properties: {}, + ), + NT4Topic( + name: 'Test/Preferences/Preference 2', + type: NT4TypeStr.kBool, + properties: {}, + ), + NT4Topic( + name: 'Test/Preferences/Preference 3', + type: NT4TypeStr.kString, + properties: {}, + ), + ], + virtualValues: { + 'Test/Preferences/Test Preference': 0, + 'Test/Preferences/Preference 1': 0.100, + 'Test/Preferences/Preference 2': false, + 'Test/Preferences/Preference 3': 'Original String' + }, + ); + }); + + test('Robot preferences from json', () { + NTWidgetModel preferencesModel = NTWidgetBuilder.buildNTModelFromJson( + ntConnection, + preferences, + 'RobotPreferences', + robotPreferencesJson, + ); + + expect(preferencesModel.type, 'RobotPreferences'); + expect(preferencesModel.runtimeType, RobotPreferencesModel); + }); + + test('Robot preferences to json', () { + RobotPreferencesModel preferencesModel = RobotPreferencesModel( + ntConnection: ntConnection, + preferences: preferences, + topic: 'Test/Preferences', + period: 0.100, + ); + + expect(preferencesModel.toJson(), robotPreferencesJson); + }); + + testWidgets('Robot preferences widget test', (widgetTester) async { + FlutterError.onError = ignoreOverflowErrors; + + NTWidgetModel preferencesModel = NTWidgetBuilder.buildNTModelFromJson( + ntConnection, + preferences, + 'RobotPreferences', + robotPreferencesJson, + ); + + await widgetTester.pumpWidget( + MaterialApp( + home: Scaffold( + body: ChangeNotifierProvider.value( + value: preferencesModel, + child: const RobotPreferences(), + ), + ), + ), + ); + + await widgetTester.pumpAndSettle(); + + expect(find.byType(TextField), findsNWidgets(5)); + + expect(find.widgetWithText(TextField, 'Test Preference'), findsOneWidget); + await widgetTester.enterText( + find.widgetWithText(TextField, 'Test Preference'), '1'); + await widgetTester.testTextInput.receiveAction(TextInputAction.done); + + expect( + ntConnection.getLastAnnouncedValue('Test/Preferences/Test Preference'), + 1); + + expect(find.widgetWithText(TextField, 'Preference 1'), findsOneWidget); + await widgetTester.enterText( + find.widgetWithText(TextField, 'Preference 1'), '0.250'); + await widgetTester.testTextInput.receiveAction(TextInputAction.done); + + expect(ntConnection.getLastAnnouncedValue('Test/Preferences/Preference 1'), + 0.250); + + expect(find.widgetWithText(TextField, 'Preference 2'), findsOneWidget); + await widgetTester.enterText( + find.widgetWithText(TextField, 'Preference 2'), 'true'); + await widgetTester.testTextInput.receiveAction(TextInputAction.done); + + expect(ntConnection.getLastAnnouncedValue('Test/Preferences/Preference 2'), + isTrue); + + expect(find.widgetWithText(TextField, 'Preference 3'), findsOneWidget); + await widgetTester.enterText( + find.widgetWithText(TextField, 'Preference 3'), 'Edited String'); + await widgetTester.testTextInput.receiveAction(TextInputAction.done); + + expect(ntConnection.getLastAnnouncedValue('Test/Preferences/Preference 3'), + 'Edited String'); + + // Searching + final searchField = find.widgetWithText(TextField, 'Search'); + expect(searchField, findsOneWidget); + + await widgetTester.enterText(searchField, 'Preference'); + await widgetTester.testTextInput.receiveAction(TextInputAction.done); + await widgetTester.pumpAndSettle(); + + expect(find.byType(TextField), findsNWidgets(5)); + + await widgetTester.enterText(searchField, 'Test'); + await widgetTester.testTextInput.receiveAction(TextInputAction.done); + await widgetTester.pumpAndSettle(); + + expect(find.byType(TextField), findsNWidgets(2)); + }); +} diff --git a/test/widgets/nt_widgets/multi-topic/split_button_chooser_test.dart b/test/widgets/nt_widgets/multi-topic/split_button_chooser_test.dart new file mode 100644 index 00000000..1dadc76b --- /dev/null +++ b/test/widgets/nt_widgets/multi-topic/split_button_chooser_test.dart @@ -0,0 +1,127 @@ +import 'package:flutter/material.dart'; + +import 'package:flutter_test/flutter_test.dart'; +import 'package:provider/provider.dart'; +import 'package:shared_preferences/shared_preferences.dart'; + +import 'package:elastic_dashboard/services/nt4_client.dart'; +import 'package:elastic_dashboard/services/nt_connection.dart'; +import 'package:elastic_dashboard/services/nt_widget_builder.dart'; +import 'package:elastic_dashboard/widgets/nt_widgets/multi-topic/split_button_chooser.dart'; +import 'package:elastic_dashboard/widgets/nt_widgets/nt_widget.dart'; +import '../../../test_util.dart'; + +void main() { + TestWidgetsFlutterBinding.ensureInitialized(); + + final Map splitButtonChooserJson = { + 'topic': 'Test/Split Button Chooser', + 'period': 0.100, + }; + + late SharedPreferences preferences; + late NTConnection ntConnection; + + setUp(() async { + SharedPreferences.setMockInitialValues({}); + preferences = await SharedPreferences.getInstance(); + + ntConnection = createMockOnlineNT4( + virtualTopics: [ + NT4Topic( + name: 'Test/Split Button Chooser/options', + type: NT4TypeStr.kStringArr, + properties: {}), + NT4Topic( + name: 'Test/Split Button Chooser/active', + type: NT4TypeStr.kString, + properties: {}), + NT4Topic( + name: 'Test/Split Button Chooser/selected', + type: NT4TypeStr.kString, + properties: {}), + NT4Topic( + name: 'Test/Split Button Chooser/default', + type: NT4TypeStr.kString, + properties: {}), + ], + virtualValues: { + 'Test/Split Button Chooser/options': ['One', 'Two', 'Three'], + 'Test/Split Button Chooser/active': 'Two', + }, + ); + }); + + test('Split button chooser from json', () { + NTWidgetModel splitButtonChooserModel = + NTWidgetBuilder.buildNTModelFromJson( + ntConnection, + preferences, + 'Split Button Chooser', + splitButtonChooserJson, + ); + + expect(splitButtonChooserModel.type, 'Split Button Chooser'); + expect(splitButtonChooserModel.runtimeType, SplitButtonChooserModel); + }); + + test('Split button chooser to json', () { + SplitButtonChooserModel splitButtonChooserModel = SplitButtonChooserModel( + ntConnection: ntConnection, + preferences: preferences, + topic: 'Test/Split Button Chooser', + period: 0.100, + ); + + expect(splitButtonChooserModel.toJson(), splitButtonChooserJson); + }); + + testWidgets('Split button chooser widget test', (widgetTester) async { + FlutterError.onError = ignoreOverflowErrors; + + NTWidgetModel splitButtonChooserModel = + NTWidgetBuilder.buildNTModelFromJson( + ntConnection, + preferences, + 'Split Button Chooser', + splitButtonChooserJson, + ); + + await widgetTester.pumpWidget( + MaterialApp( + home: Scaffold( + body: ChangeNotifierProvider.value( + value: splitButtonChooserModel, + child: const SplitButtonChooser(), + ), + ), + ), + ); + + await widgetTester.pumpAndSettle(); + + expect(find.byType(ToggleButtons), findsOneWidget); + expect(find.text('One'), findsOneWidget); + expect(find.text('Two'), findsOneWidget); + expect(find.text('Three'), findsOneWidget); + expect((splitButtonChooserModel as SplitButtonChooserModel).selectedChoice, + 'Two'); + expect(find.byIcon(Icons.check), findsOneWidget); + + await widgetTester.tap(find.text('One')); + splitButtonChooserModel.refresh(); + await widgetTester.pumpAndSettle(); + + expect(splitButtonChooserModel.selectedChoice, 'One'); + expect(find.byIcon(Icons.priority_high), findsOneWidget); + + ntConnection.updateDataFromTopicName( + splitButtonChooserModel.activeTopicName, 'One'); + + splitButtonChooserModel.refresh(); + await widgetTester.pumpAndSettle(); + + expect(find.byIcon(Icons.priority_high), findsNothing); + expect(find.byIcon(Icons.check), findsOneWidget); + }); +} diff --git a/test/widgets/nt_widgets/multi-topic/subsystem_widget_test.dart b/test/widgets/nt_widgets/multi-topic/subsystem_widget_test.dart new file mode 100644 index 00000000..eca1d9f8 --- /dev/null +++ b/test/widgets/nt_widgets/multi-topic/subsystem_widget_test.dart @@ -0,0 +1,97 @@ +import 'package:flutter/material.dart'; + +import 'package:flutter_test/flutter_test.dart'; +import 'package:provider/provider.dart'; +import 'package:shared_preferences/shared_preferences.dart'; + +import 'package:elastic_dashboard/services/nt4_client.dart'; +import 'package:elastic_dashboard/services/nt_connection.dart'; +import 'package:elastic_dashboard/services/nt_widget_builder.dart'; +import 'package:elastic_dashboard/widgets/nt_widgets/multi-topic/subsystem_widget.dart'; +import 'package:elastic_dashboard/widgets/nt_widgets/nt_widget.dart'; +import '../../../test_util.dart'; + +void main() { + TestWidgetsFlutterBinding.ensureInitialized(); + + final Map subsystemJson = { + 'topic': 'Test/Subsystem', + 'period': 0.100, + }; + + late SharedPreferences preferences; + late NTConnection ntConnection; + + setUp(() async { + SharedPreferences.setMockInitialValues({}); + preferences = await SharedPreferences.getInstance(); + + ntConnection = createMockOnlineNT4( + virtualTopics: [ + NT4Topic( + name: 'Test/Subsystem/.default', + type: NT4TypeStr.kString, + properties: {}, + ), + NT4Topic( + name: 'Test/Subsystem/.command', + type: NT4TypeStr.kString, + properties: {}, + ), + ], + virtualValues: { + 'Test/Subsystem/.command': 'TestCommand', + }, + ); + }); + + test('Subsystem model from json', () { + NTWidgetModel subsystemModel = NTWidgetBuilder.buildNTModelFromJson( + ntConnection, + preferences, + 'Subsystem', + subsystemJson, + ); + + expect(subsystemModel.type, 'Subsystem'); + expect(subsystemModel.runtimeType, SubsystemModel); + }); + + test('Subsystem model to json', () { + SubsystemModel subsystemModel = SubsystemModel( + ntConnection: ntConnection, + preferences: preferences, + topic: 'Test/Subsystem', + period: 0.100, + ); + + expect(subsystemModel.toJson(), subsystemJson); + }); + + testWidgets('Subsystem widget test', (widgetTester) async { + FlutterError.onError = ignoreOverflowErrors; + + NTWidgetModel subsystemModel = NTWidgetBuilder.buildNTModelFromJson( + ntConnection, + preferences, + 'Subsystem', + subsystemJson, + ); + + await widgetTester.pumpWidget( + MaterialApp( + home: Scaffold( + body: ChangeNotifierProvider.value( + value: subsystemModel, + child: const SubsystemWidget(), + ), + ), + ), + ); + + await widgetTester.pumpAndSettle(); + + expect(find.text('Default Command: none'), findsOneWidget); + expect(find.text('Current Command: TestCommand'), findsOneWidget); + }); +} diff --git a/test/widgets/nt_widgets/multi-topic/three_axis_accelerometer_test.dart b/test/widgets/nt_widgets/multi-topic/three_axis_accelerometer_test.dart new file mode 100644 index 00000000..4a3163ed --- /dev/null +++ b/test/widgets/nt_widgets/multi-topic/three_axis_accelerometer_test.dart @@ -0,0 +1,123 @@ +import 'package:flutter/material.dart'; + +import 'package:flutter_test/flutter_test.dart'; +import 'package:provider/provider.dart'; +import 'package:shared_preferences/shared_preferences.dart'; + +import 'package:elastic_dashboard/services/nt4_client.dart'; +import 'package:elastic_dashboard/services/nt_connection.dart'; +import 'package:elastic_dashboard/services/nt_widget_builder.dart'; +import 'package:elastic_dashboard/widgets/nt_widgets/multi-topic/three_axis_accelerometer.dart'; +import 'package:elastic_dashboard/widgets/nt_widgets/nt_widget.dart'; +import '../../../test_util.dart'; + +void main() { + TestWidgetsFlutterBinding.ensureInitialized(); + + final Map threeAxisAccelerometerJson = { + 'topic': 'Test/Three Axis Accelerometer', + 'period': 0.100, + }; + + late SharedPreferences preferences; + late NTConnection ntConnection; + + setUp(() async { + SharedPreferences.setMockInitialValues({}); + preferences = await SharedPreferences.getInstance(); + + ntConnection = createMockOnlineNT4( + virtualTopics: [ + NT4Topic( + name: 'Test/Three Axis Accelerometer/X', + type: NT4TypeStr.kFloat32, + properties: {}, + ), + NT4Topic( + name: 'Test/Three Axis Accelerometer/Y', + type: NT4TypeStr.kFloat32, + properties: {}, + ), + NT4Topic( + name: 'Test/Three Axis Accelerometer/Z', + type: NT4TypeStr.kFloat32, + properties: {}, + ), + ], + virtualValues: { + 'Test/Three Axis Accelerometer/X': 0.100, + 'Test/Three Axis Accelerometer/Y': 0.200, + 'Test/Three Axis Accelerometer/Z': 0.300, + }, + ); + }); + + test('Three axis accelerometer from json', () { + NTWidgetModel threeAxisAccelerometerModel = + NTWidgetBuilder.buildNTModelFromJson( + ntConnection, + preferences, + '3-Axis Accelerometer', + threeAxisAccelerometerJson, + ); + + expect(threeAxisAccelerometerModel.type, '3-Axis Accelerometer'); + expect( + threeAxisAccelerometerModel.runtimeType, ThreeAxisAccelerometerModel); + }); + + test('Three axis accelerometer from alias name', () { + NTWidgetModel threeAxisAccelerometerModel = + NTWidgetBuilder.buildNTModelFromJson( + ntConnection, + preferences, + '3AxisAccelerometer', + threeAxisAccelerometerJson, + ); + + expect(threeAxisAccelerometerModel.type, '3-Axis Accelerometer'); + expect( + threeAxisAccelerometerModel.runtimeType, ThreeAxisAccelerometerModel); + }); + + test('Three axis accelerometer to json', () { + ThreeAxisAccelerometerModel threeAxisAccelerometerModel = + ThreeAxisAccelerometerModel( + ntConnection: ntConnection, + preferences: preferences, + topic: 'Test/Three Axis Accelerometer', + period: 0.100, + ); + + expect(threeAxisAccelerometerModel.toJson(), threeAxisAccelerometerJson); + }); + + testWidgets('Three axis accelerometer widget test', (widgetTester) async { + FlutterError.onError = ignoreOverflowErrors; + + NTWidgetModel threeAxisAccelerometerModel = + NTWidgetBuilder.buildNTModelFromJson( + ntConnection, + preferences, + '3-Axis Accelerometer', + threeAxisAccelerometerJson, + ); + + await widgetTester.pumpWidget( + MaterialApp( + home: Scaffold( + body: ChangeNotifierProvider.value( + value: threeAxisAccelerometerModel, + child: const ThreeAxisAccelerometer(), + ), + ), + ), + ); + + await widgetTester.pumpAndSettle(); + + expect(find.text('0.10 g'), findsOneWidget); + expect(find.text('0.20 g'), findsOneWidget); + expect(find.text('0.30 g'), findsOneWidget); + }); +} diff --git a/test/widgets/nt_widgets/multi-topic/ultrasonic_test.dart b/test/widgets/nt_widgets/multi-topic/ultrasonic_test.dart new file mode 100644 index 00000000..70d1db05 --- /dev/null +++ b/test/widgets/nt_widgets/multi-topic/ultrasonic_test.dart @@ -0,0 +1,92 @@ +import 'package:flutter/material.dart'; + +import 'package:flutter_test/flutter_test.dart'; +import 'package:provider/provider.dart'; +import 'package:shared_preferences/shared_preferences.dart'; + +import 'package:elastic_dashboard/services/nt4_client.dart'; +import 'package:elastic_dashboard/services/nt_connection.dart'; +import 'package:elastic_dashboard/services/nt_widget_builder.dart'; +import 'package:elastic_dashboard/widgets/nt_widgets/multi-topic/ultrasonic.dart'; +import 'package:elastic_dashboard/widgets/nt_widgets/nt_widget.dart'; +import '../../../test_util.dart'; + +void main() { + TestWidgetsFlutterBinding.ensureInitialized(); + + final Map ultrasonicJson = { + 'topic': 'Test/Ultrasonic', + 'period': 0.100, + }; + + late SharedPreferences preferences; + late NTConnection ntConnection; + + setUp(() async { + SharedPreferences.setMockInitialValues({}); + preferences = await SharedPreferences.getInstance(); + + ntConnection = createMockOnlineNT4( + virtualTopics: [ + NT4Topic( + name: 'Test/Ultrasonic/Value', + type: NT4TypeStr.kFloat32, + properties: {}, + ), + ], + virtualValues: { + 'Test/Ultrasonic/Value': 0.12, + }, + ); + }); + + test('Ultrasonic from json', () { + NTWidgetModel ultrasonicModel = NTWidgetBuilder.buildNTModelFromJson( + ntConnection, + preferences, + 'Ultrasonic', + ultrasonicJson, + ); + + expect(ultrasonicModel.type, 'Ultrasonic'); + expect(ultrasonicModel.runtimeType, UltrasonicModel); + }); + + test('Ultrasonic to json', () { + UltrasonicModel ultrasonicModel = UltrasonicModel( + ntConnection: ntConnection, + preferences: preferences, + topic: 'Test/Ultrasonic', + period: 0.100, + ); + + expect(ultrasonicModel.toJson(), ultrasonicJson); + }); + + testWidgets('Ultrasonic widget test', (widgetTester) async { + FlutterError.onError = ignoreOverflowErrors; + + NTWidgetModel ultrasonicModel = NTWidgetBuilder.buildNTModelFromJson( + ntConnection, + preferences, + 'Ultrasonic', + ultrasonicJson, + ); + + await widgetTester.pumpWidget( + MaterialApp( + home: Scaffold( + body: ChangeNotifierProvider.value( + value: ultrasonicModel, + child: const Ultrasonic(), + ), + ), + ), + ); + + await widgetTester.pumpAndSettle(); + + expect(find.text('Range'), findsOneWidget); + expect(find.text('0.12000 in'), findsOneWidget); + }); +} diff --git a/test/widgets/nt_widgets/multi-topic/yagsl_swerve_drive_test.dart b/test/widgets/nt_widgets/multi-topic/yagsl_swerve_drive_test.dart new file mode 100644 index 00000000..8d770c1e --- /dev/null +++ b/test/widgets/nt_widgets/multi-topic/yagsl_swerve_drive_test.dart @@ -0,0 +1,90 @@ +import 'package:flutter/material.dart'; + +import 'package:flutter_test/flutter_test.dart'; +import 'package:provider/provider.dart'; +import 'package:shared_preferences/shared_preferences.dart'; + +import 'package:elastic_dashboard/services/nt_connection.dart'; +import 'package:elastic_dashboard/services/nt_widget_builder.dart'; +import 'package:elastic_dashboard/widgets/nt_widgets/multi-topic/yagsl_swerve_drive.dart'; +import 'package:elastic_dashboard/widgets/nt_widgets/nt_widget.dart'; +import '../../../test_util.dart'; + +void main() { + TestWidgetsFlutterBinding.ensureInitialized(); + + final Map yagslSwerveJson = { + 'topic': 'Test/YAGSL Swerve Drive', + 'period': 0.100, + 'show_robot_rotation': true, + 'show_desired_states': true, + }; + + late SharedPreferences preferences; + late NTConnection ntConnection; + + setUp(() async { + SharedPreferences.setMockInitialValues({}); + preferences = await SharedPreferences.getInstance(); + + ntConnection = createMockOnlineNT4(); + }); + + test('YAGSL swerve drive from json', () { + NTWidgetModel yagslSwerveModel = NTWidgetBuilder.buildNTModelFromJson( + ntConnection, + preferences, + 'YAGSL Swerve Drive', + yagslSwerveJson, + ); + + expect(yagslSwerveModel.type, 'YAGSL Swerve Drive'); + expect(yagslSwerveModel.runtimeType, YAGSLSwerveDriveModel); + + if (yagslSwerveModel is! YAGSLSwerveDriveModel) { + return; + } + + expect(yagslSwerveModel.showRobotRotation, isTrue); + expect(yagslSwerveModel.showDesiredStates, isTrue); + }); + + test('YAGSL swerve drive to json', () { + YAGSLSwerveDriveModel yagslSwerveModel = YAGSLSwerveDriveModel( + ntConnection: ntConnection, + preferences: preferences, + topic: 'Test/YAGSL Swerve Drive', + period: 0.100, + showRobotRotation: true, + showDesiredStates: true, + ); + + expect(yagslSwerveModel.toJson(), yagslSwerveJson); + }); + + testWidgets('YAGSL swerve drive widget test', (widgetTester) async { + FlutterError.onError = ignoreOverflowErrors; + + NTWidgetModel yagslSwerveModel = NTWidgetBuilder.buildNTModelFromJson( + ntConnection, + preferences, + 'YAGSL Swerve Drive', + yagslSwerveJson, + ); + + await widgetTester.pumpWidget( + MaterialApp( + home: Scaffold( + body: ChangeNotifierProvider.value( + value: yagslSwerveModel, + child: const YAGSLSwerveDrive(), + ), + ), + ), + ); + + await widgetTester.pumpAndSettle(); + + expect(find.byType(CustomPaint), findsWidgets); + }); +} diff --git a/test/widgets/nt_widgets/single-topic/boolean_box_test.dart b/test/widgets/nt_widgets/single-topic/boolean_box_test.dart new file mode 100644 index 00000000..e7530820 --- /dev/null +++ b/test/widgets/nt_widgets/single-topic/boolean_box_test.dart @@ -0,0 +1,150 @@ +import 'package:flutter/material.dart'; + +import 'package:flutter_test/flutter_test.dart'; +import 'package:provider/provider.dart'; +import 'package:shared_preferences/shared_preferences.dart'; + +import 'package:elastic_dashboard/services/nt4_client.dart'; +import 'package:elastic_dashboard/services/nt_connection.dart'; +import 'package:elastic_dashboard/services/nt_widget_builder.dart'; +import 'package:elastic_dashboard/widgets/nt_widgets/nt_widget.dart'; +import 'package:elastic_dashboard/widgets/nt_widgets/single_topic/boolean_box.dart'; +import '../../../test_util.dart'; + +void main() { + TestWidgetsFlutterBinding.ensureInitialized(); + + final Map booleanBoxJson = { + 'topic': 'Test/Boolean Value', + 'period': 0.100, + 'data_type': 'boolean', + 'true_color': Colors.green.value, + 'false_color': Colors.red.value, + 'true_icon': 'None', + 'false_icon': 'None', + }; + + late SharedPreferences preferences; + late NTConnection ntConnection; + + Finder findColor(Color color) => find.byWidgetPredicate( + (widget) => + widget is Container && + widget.decoration is BoxDecoration && + (widget.decoration as BoxDecoration).color != null && + (widget.decoration as BoxDecoration).color!.value == color.value, + ); + + setUp(() async { + SharedPreferences.setMockInitialValues({}); + preferences = await SharedPreferences.getInstance(); + + ntConnection = createMockOnlineNT4( + virtualTopics: [ + NT4Topic( + name: 'Test/Boolean Value', + type: NT4TypeStr.kBool, + properties: {}, + ), + ], + virtualValues: { + 'Test/Boolean Value': false, + }, + ); + }); + + test('Boolean box from json', () { + NTWidgetModel booleanBoxModel = NTWidgetBuilder.buildNTModelFromJson( + ntConnection, + preferences, + 'Boolean Box', + booleanBoxJson, + ); + + expect(booleanBoxModel.type, 'Boolean Box'); + expect(booleanBoxModel.runtimeType, BooleanBoxModel); + expect( + booleanBoxModel.getAvailableDisplayTypes(), + unorderedEquals([ + 'Boolean Box', + 'Toggle Switch', + 'Toggle Button', + 'Text Display', + ])); + + if (booleanBoxModel is! BooleanBoxModel) { + return; + } + + expect(booleanBoxModel.trueColor, Color(Colors.green.value)); + expect(booleanBoxModel.falseColor, Color(Colors.red.value)); + expect(booleanBoxModel.trueIcon, 'None'); + expect(booleanBoxModel.falseIcon, 'None'); + }); + + test('Boolean box to json', () { + BooleanBoxModel booleanBoxModel = BooleanBoxModel( + ntConnection: ntConnection, + preferences: preferences, + topic: 'Test/Boolean Value', + dataType: 'boolean', + period: 0.100, + trueColor: Colors.green, + falseColor: Colors.red, + trueIcon: 'None', + falseIcon: 'None', + ); + + expect(booleanBoxModel.toJson(), booleanBoxJson); + }); + + testWidgets('Boolean box widget test', (widgetTester) async { + FlutterError.onError = ignoreOverflowErrors; + + NTWidgetModel booleanBoxModel = NTWidgetBuilder.buildNTModelFromJson( + ntConnection, + preferences, + 'Boolean Box', + booleanBoxJson, + ); + + await widgetTester.pumpWidget( + MaterialApp( + home: Scaffold( + body: ChangeNotifierProvider.value( + value: booleanBoxModel as BooleanBoxModel, + child: const BooleanBox(), + ), + ), + ), + ); + + await widgetTester.pumpAndSettle(); + + expect(findColor(Colors.green), findsNothing); + expect(findColor(Colors.red), findsOneWidget); + + ntConnection.updateDataFromTopicName('Test/Boolean Value', true); + booleanBoxModel.refresh(); + await widgetTester.pumpAndSettle(); + + expect(findColor(Colors.green), findsOneWidget); + expect(findColor(Colors.red), findsNothing); + + booleanBoxModel.trueIcon = 'Checkmark'; + await widgetTester.pumpAndSettle(); + + expect(findColor(Colors.green), findsNothing); + expect(find.byIcon(Icons.check), findsOneWidget); + + ntConnection.updateDataFromTopicName('Test/Boolean Value', false); + booleanBoxModel.falseIcon = 'X'; + await widgetTester.pumpAndSettle(); + + expect(find.byIcon(Icons.clear), findsOneWidget); + + booleanBoxModel.falseIcon = 'Exclamation Point'; + await widgetTester.pumpAndSettle(); + expect(find.byIcon(Icons.priority_high), findsOneWidget); + }); +} diff --git a/test/widgets/nt_widgets/single-topic/graph_test.dart b/test/widgets/nt_widgets/single-topic/graph_test.dart new file mode 100644 index 00000000..08d8e9c9 --- /dev/null +++ b/test/widgets/nt_widgets/single-topic/graph_test.dart @@ -0,0 +1,121 @@ +import 'package:flutter/material.dart'; + +import 'package:flutter_test/flutter_test.dart'; +import 'package:provider/provider.dart'; +import 'package:shared_preferences/shared_preferences.dart'; +import 'package:syncfusion_flutter_charts/charts.dart'; + +import 'package:elastic_dashboard/services/nt4_client.dart'; +import 'package:elastic_dashboard/services/nt_connection.dart'; +import 'package:elastic_dashboard/services/nt_widget_builder.dart'; +import 'package:elastic_dashboard/widgets/nt_widgets/nt_widget.dart'; +import 'package:elastic_dashboard/widgets/nt_widgets/single_topic/graph.dart'; +import '../../../test_util.dart'; + +void main() { + TestWidgetsFlutterBinding.ensureInitialized(); + + final Map graphJson = { + 'topic': 'Test/Double Value', + 'data_type': 'double', + 'period': 0.100, + 'time_displayed': 10.0, + 'max_value': 1.0, + 'color': Colors.green.value, + 'line_width': 3.0, + }; + + late SharedPreferences preferences; + late NTConnection ntConnection; + + setUp(() async { + SharedPreferences.setMockInitialValues({}); + preferences = await SharedPreferences.getInstance(); + + ntConnection = createMockOnlineNT4( + virtualTopics: [ + NT4Topic( + name: 'Test/Double Value', + type: NT4TypeStr.kFloat64, + properties: {}, + ), + ], + virtualValues: { + 'Test/Double Value': 0.0, + }, + ); + }); + + test('Graph from json', () { + NTWidgetModel graphModel = NTWidgetBuilder.buildNTModelFromJson( + ntConnection, + preferences, + 'Graph', + graphJson, + ); + + expect(graphModel.type, 'Graph'); + expect(graphModel.runtimeType, GraphModel); + expect( + graphModel.getAvailableDisplayTypes(), + unorderedEquals([ + 'Text Display', + 'Number Bar', + 'Number Slider', + 'Graph', + 'Voltage View', + 'Radial Gauge', + 'Match Time', + ])); + + if (graphModel is! GraphModel) { + return; + } + + expect(graphModel.timeDisplayed, 10.0); + expect(graphModel.minValue, isNull); + expect(graphModel.maxValue, 1.0); + expect(graphModel.mainColor, Color(Colors.green.value)); + expect(graphModel.lineWidth, 3.0); + }); + + test('Graph model to json', () { + GraphModel graphModel = GraphModel( + ntConnection: ntConnection, + preferences: preferences, + topic: 'Test/Double Value', + dataType: 'double', + period: 0.100, + timeDisplayed: 10.0, + maxValue: 1.0, + mainColor: Colors.green, + lineWidth: 3.0, + ); + + expect(graphModel.toJson(), graphJson); + }); + + testWidgets('Graph widget test', (widgetTester) async { + FlutterError.onError = ignoreOverflowErrors; + + NTWidgetModel graphModel = NTWidgetBuilder.buildNTModelFromJson( + ntConnection, + preferences, + 'Graph', + graphJson, + ); + + await widgetTester.pumpWidget( + MaterialApp( + home: Scaffold( + body: ChangeNotifierProvider.value( + value: graphModel, + child: const GraphWidget(), + ), + ), + ), + ); + + expect(find.byType(SfCartesianChart), findsOneWidget); + }); +} diff --git a/test/widgets/nt_widgets/single-topic/match_time_test.dart b/test/widgets/nt_widgets/single-topic/match_time_test.dart new file mode 100644 index 00000000..05f29a6a --- /dev/null +++ b/test/widgets/nt_widgets/single-topic/match_time_test.dart @@ -0,0 +1,156 @@ +import 'package:flutter/material.dart'; + +import 'package:flutter_test/flutter_test.dart'; +import 'package:provider/provider.dart'; +import 'package:shared_preferences/shared_preferences.dart'; + +import 'package:elastic_dashboard/services/nt4_client.dart'; +import 'package:elastic_dashboard/services/nt_connection.dart'; +import 'package:elastic_dashboard/services/nt_widget_builder.dart'; +import 'package:elastic_dashboard/widgets/nt_widgets/nt_widget.dart'; +import 'package:elastic_dashboard/widgets/nt_widgets/single_topic/match_time.dart'; +import '../../../test_util.dart'; + +void main() { + TestWidgetsFlutterBinding.ensureInitialized(); + + final Map matchTimeJson = { + 'topic': 'Test/Double Value', + 'data_type': 'double', + 'period': 0.100, + 'time_display_mode': 'Minutes and Seconds', + 'red_start_time': 15, + 'yellow_start_time': 25.0, + }; + + Finder coloredText(String text, Color color) => find.byWidgetPredicate( + (widget) => + widget is Text && + widget.data == text && + widget.style != null && + widget.style!.color != null && + widget.style!.color!.value == color.value, + ); + + late SharedPreferences preferences; + late NTConnection ntConnection; + + setUp(() async { + SharedPreferences.setMockInitialValues({}); + preferences = await SharedPreferences.getInstance(); + + ntConnection = createMockOnlineNT4( + virtualTopics: [ + NT4Topic( + name: 'Test/Double Value', + type: NT4TypeStr.kFloat64, + properties: {}, + ), + ], + virtualValues: { + 'Test/Double Value': 96.0, + }, + ); + }); + + test('Match time from json', () { + NTWidgetModel matchTimeModel = NTWidgetBuilder.buildNTModelFromJson( + ntConnection, + preferences, + 'Match Time', + matchTimeJson, + ); + + expect(matchTimeModel.type, 'Match Time'); + expect(matchTimeModel.runtimeType, MatchTimeModel); + expect( + matchTimeModel.getAvailableDisplayTypes(), + unorderedEquals([ + 'Text Display', + 'Number Bar', + 'Number Slider', + 'Graph', + 'Voltage View', + 'Radial Gauge', + 'Match Time', + ])); + + if (matchTimeModel is! MatchTimeModel) { + return; + } + + expect(matchTimeModel.timeDisplayMode, 'Minutes and Seconds'); + expect(matchTimeModel.redStartTime, 15); + expect(matchTimeModel.yellowStartTime, 25); + }); + + test('Match time to json', () { + MatchTimeModel matchTimeModel = MatchTimeModel( + ntConnection: ntConnection, + preferences: preferences, + topic: 'Test/Double Value', + dataType: 'double', + period: 0.100, + timeDisplayMode: 'Minutes and Seconds', + redStartTime: 15, + yellowStartTime: 25, + ); + + expect(matchTimeModel.toJson(), matchTimeJson); + }); + + testWidgets('Match time widget test', (widgetTester) async { + FlutterError.onError = ignoreOverflowErrors; + + NTWidgetModel matchTimeModel = NTWidgetBuilder.buildNTModelFromJson( + ntConnection, + preferences, + 'Match Time', + matchTimeJson, + ); + + await widgetTester.pumpWidget( + MaterialApp( + home: Scaffold( + body: ChangeNotifierProvider.value( + value: matchTimeModel as MatchTimeModel, + child: const MatchTimeWidget(), + ), + ), + ), + ); + + await widgetTester.pumpAndSettle(); + + expect(find.text('1:36'), findsOneWidget); + expect(coloredText('1:36', Colors.blue), findsOneWidget); + + ntConnection.updateDataFromTopicName('Test/Double Value', 55.0); + matchTimeModel.refresh(); + await widgetTester.pumpAndSettle(); + + expect(find.text('0:55'), findsOneWidget); + expect(coloredText('0:55', Colors.green), findsOneWidget); + + ntConnection.updateDataFromTopicName('Test/Double Value', 22.0); + matchTimeModel.refresh(); + await widgetTester.pumpAndSettle(); + + expect(find.text('0:22'), findsOneWidget); + expect(coloredText('0:22', Colors.yellow), findsOneWidget); + + ntConnection.updateDataFromTopicName('Test/Double Value', 13.0); + matchTimeModel.refresh(); + await widgetTester.pumpAndSettle(); + + expect(find.text('0:13'), findsOneWidget); + expect(coloredText('0:13', Colors.red), findsOneWidget); + + ntConnection.updateDataFromTopicName('Test/Double Value', 96.0); + matchTimeModel.timeDisplayMode = 'Seconds Only'; + await widgetTester.pumpAndSettle(); + + expect(find.text('96'), findsOneWidget); + expect(coloredText('96', Colors.blue), findsOneWidget); + }); +} diff --git a/test/widgets/nt_widgets/single-topic/multi_color_view_test.dart b/test/widgets/nt_widgets/single-topic/multi_color_view_test.dart new file mode 100644 index 00000000..a253570c --- /dev/null +++ b/test/widgets/nt_widgets/single-topic/multi_color_view_test.dart @@ -0,0 +1,215 @@ +import 'package:flutter/material.dart'; + +import 'package:collection/collection.dart'; +import 'package:flutter_colorpicker/flutter_colorpicker.dart'; +import 'package:flutter_test/flutter_test.dart'; +import 'package:provider/provider.dart'; +import 'package:shared_preferences/shared_preferences.dart'; + +import 'package:elastic_dashboard/services/nt4_client.dart'; +import 'package:elastic_dashboard/services/nt_connection.dart'; +import 'package:elastic_dashboard/services/nt_widget_builder.dart'; +import 'package:elastic_dashboard/widgets/nt_widgets/nt_widget.dart'; +import 'package:elastic_dashboard/widgets/nt_widgets/single_topic/multi_color_view.dart'; +import '../../../test_util.dart'; + +void main() { + TestWidgetsFlutterBinding.ensureInitialized(); + + final Map multiColorViewJson = { + 'topic': 'Test/String Array', + 'data_type': 'string[]', + 'period': 0.100, + }; + + Finder findGradient(List expectedColors) => + find.byWidgetPredicate((widget) => + widget is Container && + widget.decoration is BoxDecoration && + (widget.decoration as BoxDecoration).gradient != null && + (widget.decoration as BoxDecoration) + .gradient! + .colors + .equals(expectedColors)); + + late SharedPreferences preferences; + late NTConnection ntConnection; + + setUp(() async { + SharedPreferences.setMockInitialValues({}); + preferences = await SharedPreferences.getInstance(); + + ntConnection = createMockOnlineNT4( + virtualTopics: [ + NT4Topic( + name: 'Test/String Array', + type: NT4TypeStr.kStringArr, + properties: {}, + ), + ], + virtualValues: { + 'Test/String Array': [ + Colors.red.toHexString(includeHashSign: true), + Colors.orange.toHexString(includeHashSign: true), + Colors.yellow.toHexString(includeHashSign: true), + Colors.green.toHexString(includeHashSign: true), + Colors.blue.toHexString(includeHashSign: true), + Colors.purple.toHexString(includeHashSign: true), + ], + }, + ); + }); + + test('Multi color view from json', () { + NTWidgetModel multiColorViewModel = NTWidgetBuilder.buildNTModelFromJson( + ntConnection, + preferences, + 'Multi Color View', + multiColorViewJson, + ); + + expect(multiColorViewModel.type, 'Multi Color View'); + expect(multiColorViewModel.runtimeType, NTWidgetModel); + expect( + multiColorViewModel.getAvailableDisplayTypes(), + unorderedEquals([ + 'Multi Color View', + 'Text Display', + ])); + }); + + test('Multi color view to json', () { + NTWidgetModel multiColorViewModel = NTWidgetModel.createDefault( + ntConnection: ntConnection, + preferences: preferences, + type: 'Multi Color View', + topic: 'Test/String Array', + dataType: 'string[]', + period: 0.100, + ); + + expect(multiColorViewModel.toJson(), multiColorViewJson); + }); + + testWidgets('Multi color view widget test full gradient', + (widgetTester) async { + FlutterError.onError = ignoreOverflowErrors; + + NTWidgetModel multiColorViewModel = NTWidgetBuilder.buildNTModelFromJson( + ntConnection, + preferences, + 'Multi Color View', + multiColorViewJson, + ); + + await widgetTester.pumpWidget( + MaterialApp( + home: Scaffold( + body: ChangeNotifierProvider.value( + value: multiColorViewModel, + child: const MultiColorView(), + ), + ), + ), + ); + + await widgetTester.pumpAndSettle(); + + List expectedColors = [ + Color(Colors.red.value), + Color(Colors.orange.value), + Color(Colors.yellow.value), + Color(Colors.green.value), + Color(Colors.blue.value), + Color(Colors.purple.value), + ]; + + expect(findGradient(expectedColors), findsOneWidget); + }); + + testWidgets('Multi color view widget test one color', (widgetTester) async { + FlutterError.onError = ignoreOverflowErrors; + + NTWidgetModel multiColorViewModel = NTWidgetBuilder.buildNTModelFromJson( + createMockOnlineNT4( + virtualTopics: [ + NT4Topic( + name: 'Test/String Array', + type: NT4TypeStr.kStringArr, + properties: {}, + ), + ], + virtualValues: { + 'Test/String Array': [ + Colors.red.toHexString(includeHashSign: true), + ], + }, + ), + preferences, + 'Multi Color View', + multiColorViewJson, + ); + + await widgetTester.pumpWidget( + MaterialApp( + home: Scaffold( + body: ChangeNotifierProvider.value( + value: multiColorViewModel, + child: const MultiColorView(), + ), + ), + ), + ); + + await widgetTester.pumpAndSettle(); + + expect( + findGradient([ + Color(Colors.red.value), + Color(Colors.red.value), + ]), + findsOneWidget); + }); + + testWidgets('Multi color view widget test no colors', (widgetTester) async { + FlutterError.onError = ignoreOverflowErrors; + + NTWidgetModel multiColorViewModel = NTWidgetBuilder.buildNTModelFromJson( + createMockOnlineNT4( + virtualTopics: [ + NT4Topic( + name: 'Test/String Array', + type: NT4TypeStr.kStringArr, + properties: {}, + ), + ], + virtualValues: { + 'Test/String Array': [], + }, + ), + preferences, + 'Multi Color View', + multiColorViewJson, + ); + + await widgetTester.pumpWidget( + MaterialApp( + home: Scaffold( + body: ChangeNotifierProvider.value( + value: multiColorViewModel, + child: const MultiColorView(), + ), + ), + ), + ); + + await widgetTester.pumpAndSettle(); + + expect( + findGradient([ + Color(Colors.transparent.value), + Color(Colors.transparent.value), + ]), + findsOneWidget); + }); +} diff --git a/test/widgets/nt_widgets/single-topic/number_bar_test.dart b/test/widgets/nt_widgets/single-topic/number_bar_test.dart new file mode 100644 index 00000000..6e2337f6 --- /dev/null +++ b/test/widgets/nt_widgets/single-topic/number_bar_test.dart @@ -0,0 +1,212 @@ +import 'package:flutter/material.dart'; + +import 'package:flutter_test/flutter_test.dart'; +import 'package:provider/provider.dart'; +import 'package:shared_preferences/shared_preferences.dart'; +import 'package:syncfusion_flutter_gauges/gauges.dart'; + +import 'package:elastic_dashboard/services/nt4_client.dart'; +import 'package:elastic_dashboard/services/nt_connection.dart'; +import 'package:elastic_dashboard/services/nt_widget_builder.dart'; +import 'package:elastic_dashboard/widgets/nt_widgets/nt_widget.dart'; +import 'package:elastic_dashboard/widgets/nt_widgets/single_topic/number_bar.dart'; +import '../../../test_util.dart'; + +void main() { + TestWidgetsFlutterBinding.ensureInitialized(); + + final Map numberBarJson = { + 'topic': 'Test/Double Value', + 'data_type': 'double', + 'period': 0.100, + 'min_value': -5.0, + 'max_value': 5.0, + 'inverted': false, + 'orientation': 'horizontal', + }; + + late SharedPreferences preferences; + late NTConnection ntConnection; + + setUp(() async { + SharedPreferences.setMockInitialValues({}); + preferences = await SharedPreferences.getInstance(); + + ntConnection = createMockOnlineNT4( + virtualTopics: [ + NT4Topic( + name: 'Test/Double Value', + type: NT4TypeStr.kFloat64, + properties: {}, + ), + ], + virtualValues: { + 'Test/Double Value': -1.0, + }, + ); + }); + + test('Number bar from json', () { + NTWidgetModel numberBarModel = NTWidgetBuilder.buildNTModelFromJson( + ntConnection, + preferences, + 'Number Bar', + numberBarJson, + ); + + expect(numberBarModel.type, 'Number Bar'); + expect(numberBarModel.runtimeType, NumberBarModel); + expect( + numberBarModel.getAvailableDisplayTypes(), + unorderedEquals([ + 'Text Display', + 'Number Bar', + 'Number Slider', + 'Graph', + 'Voltage View', + 'Radial Gauge', + 'Match Time', + ])); + + if (numberBarModel is! NumberBarModel) { + return; + } + + expect(numberBarModel.minValue, -5.0); + expect(numberBarModel.maxValue, 5.0); + expect(numberBarModel.divisions, isNull); + expect(numberBarModel.inverted, isFalse); + expect(numberBarModel.orientation, 'horizontal'); + }); + + test('Number bar to json', () { + NumberBarModel numberBarModel = NumberBarModel( + ntConnection: ntConnection, + preferences: preferences, + topic: 'Test/Double Value', + dataType: 'double', + period: 0.100, + minValue: -5.0, + maxValue: 5.0, + divisions: null, + inverted: false, + orientation: 'horizontal', + ); + + expect(numberBarModel.toJson(), numberBarJson); + }); + + testWidgets('Number bar widget test horizontal', (widgetTester) async { + FlutterError.onError = ignoreOverflowErrors; + + NTWidgetModel numberBarModel = NumberBarModel( + ntConnection: ntConnection, + preferences: preferences, + topic: 'Test/Double Value', + dataType: 'double', + period: 0.100, + minValue: -5.0, + maxValue: 5.0, + divisions: null, + inverted: false, + orientation: 'horizontal', + ); + + await widgetTester.pumpWidget( + MaterialApp( + home: Scaffold( + body: ChangeNotifierProvider.value( + value: numberBarModel, + child: const NumberBar(), + ), + ), + ), + ); + + await widgetTester.pumpAndSettle(); + + expect(find.text('-1.00'), findsOneWidget); + expect(find.byType(SfLinearGauge), findsOneWidget); + + expect( + (find.byType(SfLinearGauge).evaluate().first.widget as SfLinearGauge) + .orientation, + LinearGaugeOrientation.horizontal); + }); + + testWidgets('Number bar widget test vertical', (widgetTester) async { + FlutterError.onError = ignoreOverflowErrors; + + NTWidgetModel numberBarModel = NumberBarModel( + ntConnection: ntConnection, + preferences: preferences, + topic: 'Test/Double Value', + dataType: 'double', + period: 0.100, + minValue: -5.0, + maxValue: 5.0, + divisions: null, + inverted: false, + orientation: 'vertical', + ); + + await widgetTester.pumpWidget( + MaterialApp( + home: Scaffold( + body: ChangeNotifierProvider.value( + value: numberBarModel, + child: const NumberBar(), + ), + ), + ), + ); + + await widgetTester.pumpAndSettle(); + + expect(find.text('-1.00'), findsOneWidget); + expect(find.byType(SfLinearGauge), findsOneWidget); + + expect( + (find.byType(SfLinearGauge).evaluate().first.widget as SfLinearGauge) + .orientation, + LinearGaugeOrientation.vertical); + }); + + testWidgets('Number bar widget test with divisions', (widgetTester) async { + FlutterError.onError = ignoreOverflowErrors; + + NTWidgetModel numberBarModel = NumberBarModel( + ntConnection: ntConnection, + preferences: preferences, + topic: 'Test/Double Value', + dataType: 'double', + period: 0.100, + minValue: -5.0, + maxValue: 5.0, + divisions: 11, + inverted: false, + orientation: 'horizontal', + ); + + await widgetTester.pumpWidget( + MaterialApp( + home: Scaffold( + body: ChangeNotifierProvider.value( + value: numberBarModel, + child: const NumberBar(), + ), + ), + ), + ); + + await widgetTester.pumpAndSettle(); + + expect(find.text('-1.00'), findsOneWidget); + expect(find.byType(SfLinearGauge), findsOneWidget); + + expect( + (find.byType(SfLinearGauge).evaluate().first.widget as SfLinearGauge) + .interval, + 1.0); + }); +} diff --git a/test/widgets/nt_widgets/single-topic/number_slider_test.dart b/test/widgets/nt_widgets/single-topic/number_slider_test.dart new file mode 100644 index 00000000..2fecbb1d --- /dev/null +++ b/test/widgets/nt_widgets/single-topic/number_slider_test.dart @@ -0,0 +1,205 @@ +import 'package:flutter/material.dart'; + +import 'package:flutter_test/flutter_test.dart'; +import 'package:provider/provider.dart'; +import 'package:shared_preferences/shared_preferences.dart'; +import 'package:syncfusion_flutter_gauges/gauges.dart'; + +import 'package:elastic_dashboard/services/nt4_client.dart'; +import 'package:elastic_dashboard/services/nt_connection.dart'; +import 'package:elastic_dashboard/services/nt_widget_builder.dart'; +import 'package:elastic_dashboard/widgets/nt_widgets/nt_widget.dart'; +import 'package:elastic_dashboard/widgets/nt_widgets/single_topic/number_slider.dart'; +import '../../../test_util.dart'; + +void main() { + TestWidgetsFlutterBinding.ensureInitialized(); + + final Map numberSliderJson = { + 'topic': 'Test/Double Value', + 'data_type': 'double', + 'period': 0.100, + 'min_value': -5.0, + 'max_value': 5.0, + 'divisions': 5, + 'update_continuously': true, + }; + + late SharedPreferences preferences; + late NTConnection ntConnection; + + setUp(() async { + SharedPreferences.setMockInitialValues({}); + preferences = await SharedPreferences.getInstance(); + + ntConnection = createMockOnlineNT4( + virtualTopics: [ + NT4Topic( + name: 'Test/Double Value', + type: NT4TypeStr.kFloat64, + properties: {}, + ), + ], + virtualValues: { + 'Test/Double Value': -1.0, + }, + ); + }); + + test('Number slider from json', () { + NTWidgetModel numberSliderModel = NTWidgetBuilder.buildNTModelFromJson( + ntConnection, + preferences, + 'Number Slider', + numberSliderJson, + ); + + expect(numberSliderModel.type, 'Number Slider'); + expect(numberSliderModel.runtimeType, NumberSliderModel); + expect( + numberSliderModel.getAvailableDisplayTypes(), + unorderedEquals([ + 'Text Display', + 'Number Bar', + 'Number Slider', + 'Graph', + 'Voltage View', + 'Radial Gauge', + 'Match Time', + ])); + + if (numberSliderModel is! NumberSliderModel) { + return; + } + + expect(numberSliderModel.minValue, -5.0); + expect(numberSliderModel.maxValue, 5.0); + expect(numberSliderModel.divisions, 5); + expect(numberSliderModel.updateContinuously, isTrue); + }); + + test('Number slider to json', () { + NumberSliderModel numberSliderModel = NumberSliderModel( + ntConnection: ntConnection, + preferences: preferences, + topic: 'Test/Double Value', + dataType: 'double', + period: 0.100, + minValue: -5.0, + maxValue: 5.0, + divisions: 5, + updateContinuously: true, + ); + + expect(numberSliderModel.toJson(), numberSliderJson); + }); + + testWidgets('Number slider widget test continuous update', + (widgetTester) async { + FlutterError.onError = ignoreOverflowErrors; + + NTWidgetModel numberSliderModel = NTWidgetBuilder.buildNTModelFromJson( + ntConnection, + preferences, + 'Number Slider', + numberSliderJson, + ); + + await widgetTester.pumpWidget( + MaterialApp( + home: Scaffold( + body: ChangeNotifierProvider.value( + value: numberSliderModel, + child: const NumberSlider(), + ), + ), + ), + ); + + await widgetTester.pumpAndSettle(); + + expect(find.text('-1.00'), findsOneWidget); + expect(find.byType(SfLinearGauge), findsOneWidget); + expect(find.byType(LinearShapePointer), findsOneWidget); + + Future pointerDrag = widgetTester.timedDrag( + find.byType(LinearShapePointer), + const Offset(100.0, 0.0), + const Duration(seconds: 1), + ); + + // Stupid workaround since expect can't be used during a drag + bool? draggingDuringDrag; + Object? valueDuringDrag; + + Future.delayed(const Duration(milliseconds: 500), () { + draggingDuringDrag = (numberSliderModel as NumberSliderModel).dragging; + valueDuringDrag = ntConnection.getLastAnnouncedValue('Test/Double Value'); + }); + + await pointerDrag; + + expect(draggingDuringDrag, isTrue); + expect(valueDuringDrag, isNot(-1.0)); + + expect(ntConnection.getLastAnnouncedValue('Test/Double Value'), + greaterThan(0.0)); + }); + + testWidgets('Number slider widget test non-continuous update', + (widgetTester) async { + FlutterError.onError = ignoreOverflowErrors; + + NumberSliderModel numberSliderModel = NumberSliderModel( + ntConnection: ntConnection, + preferences: preferences, + topic: 'Test/Double Value', + dataType: 'double', + period: 0.100, + minValue: -5.0, + maxValue: 5.0, + divisions: 5, + updateContinuously: false, + ); + + await widgetTester.pumpWidget( + MaterialApp( + home: Scaffold( + body: ChangeNotifierProvider.value( + value: numberSliderModel, + child: const NumberSlider(), + ), + ), + ), + ); + + await widgetTester.pumpAndSettle(); + + expect(find.text('-1.00'), findsOneWidget); + expect(find.byType(SfLinearGauge), findsOneWidget); + expect(find.byType(LinearShapePointer), findsOneWidget); + + Future pointerDrag = widgetTester.timedDrag( + find.byType(LinearShapePointer), + const Offset(100.0, 0.0), + const Duration(seconds: 1), + ); + + // Stupid workaround since expect can't be used during a drag + bool? draggingDuringDrag; + Object? valueDuringDrag; + + Future.delayed(const Duration(milliseconds: 500), () { + draggingDuringDrag = numberSliderModel.dragging; + valueDuringDrag = ntConnection.getLastAnnouncedValue('Test/Double Value'); + }); + + await pointerDrag; + + expect(draggingDuringDrag, isTrue); + expect(valueDuringDrag, -1.0); + + expect(ntConnection.getLastAnnouncedValue('Test/Double Value'), + greaterThan(0.0)); + }); +} diff --git a/test/widgets/nt_widgets/single-topic/radial_gauge_test.dart b/test/widgets/nt_widgets/single-topic/radial_gauge_test.dart new file mode 100644 index 00000000..12a645ec --- /dev/null +++ b/test/widgets/nt_widgets/single-topic/radial_gauge_test.dart @@ -0,0 +1,218 @@ +import 'package:flutter/material.dart'; + +import 'package:flutter_test/flutter_test.dart'; +import 'package:provider/provider.dart'; +import 'package:shared_preferences/shared_preferences.dart'; +import 'package:syncfusion_flutter_gauges/gauges.dart'; + +import 'package:elastic_dashboard/services/nt4_client.dart'; +import 'package:elastic_dashboard/services/nt_connection.dart'; +import 'package:elastic_dashboard/services/nt_widget_builder.dart'; +import 'package:elastic_dashboard/widgets/nt_widgets/nt_widget.dart'; +import 'package:elastic_dashboard/widgets/nt_widgets/single_topic/radial_gauge.dart'; +import '../../../test_util.dart'; + +void main() { + TestWidgetsFlutterBinding.ensureInitialized(); + + final Map radialGaugeJson = { + 'topic': 'Test/Double Value', + 'data_type': 'double', + 'period': 0.100, + 'start_angle': -140.0, + 'end_angle': 140.0, + 'min_value': -1.0, + 'max_value': 1.0, + 'number_of_labels': 10, + 'wrap_value': false, + 'show_pointer': true, + 'show_ticks': true, + }; + + late SharedPreferences preferences; + late NTConnection ntConnection; + + setUp(() async { + SharedPreferences.setMockInitialValues({}); + preferences = await SharedPreferences.getInstance(); + + ntConnection = createMockOnlineNT4( + virtualTopics: [ + NT4Topic( + name: 'Test/Double Value', + type: NT4TypeStr.kFloat64, + properties: {}, + ), + ], + virtualValues: { + 'Test/Double Value': -0.50, + }, + ); + }); + + test('Radial gauge from json', () { + NTWidgetModel radialGaugeModel = NTWidgetBuilder.buildNTModelFromJson( + ntConnection, + preferences, + 'Radial Gauge', + radialGaugeJson, + ); + + expect(radialGaugeModel.type, 'Radial Gauge'); + expect(radialGaugeModel.runtimeType, RadialGaugeModel); + expect( + radialGaugeModel.getAvailableDisplayTypes(), + unorderedEquals([ + 'Text Display', + 'Number Bar', + 'Number Slider', + 'Graph', + 'Voltage View', + 'Radial Gauge', + 'Match Time', + ])); + + if (radialGaugeModel is! RadialGaugeModel) { + return; + } + + expect(radialGaugeModel.startAngle, -140.0); + expect(radialGaugeModel.endAngle, 140.0); + expect(radialGaugeModel.minValue, -1.0); + expect(radialGaugeModel.maxValue, 1.0); + expect(radialGaugeModel.numberOfLabels, 10); + expect(radialGaugeModel.wrapValue, isFalse); + expect(radialGaugeModel.showPointer, isTrue); + expect(radialGaugeModel.showTicks, isTrue); + }); + + test('Radial gauge from alias name', () { + NTWidgetModel radialGaugeModel = NTWidgetBuilder.buildNTModelFromJson( + ntConnection, + preferences, + 'Simple Dial', + radialGaugeJson, + ); + + expect(radialGaugeModel.type, 'Radial Gauge'); + expect(radialGaugeModel.runtimeType, RadialGaugeModel); + expect( + radialGaugeModel.getAvailableDisplayTypes(), + unorderedEquals([ + 'Text Display', + 'Number Bar', + 'Number Slider', + 'Graph', + 'Voltage View', + 'Radial Gauge', + 'Match Time', + ])); + + if (radialGaugeModel is! RadialGaugeModel) { + return; + } + + expect(radialGaugeModel.startAngle, -140.0); + expect(radialGaugeModel.endAngle, 140.0); + expect(radialGaugeModel.minValue, -1.0); + expect(radialGaugeModel.maxValue, 1.0); + expect(radialGaugeModel.numberOfLabels, 10); + expect(radialGaugeModel.wrapValue, isFalse); + expect(radialGaugeModel.showPointer, isTrue); + expect(radialGaugeModel.showTicks, isTrue); + }); + + test('Radial gauge to json', () { + RadialGaugeModel radialGaugeModel = RadialGaugeModel( + ntConnection: ntConnection, + preferences: preferences, + topic: 'Test/Double Value', + dataType: 'double', + period: 0.100, + startAngle: -140.0, + endAngle: 140.0, + minValue: -1.0, + maxValue: 1.0, + numberOfLabels: 10, + wrapValue: false, + showPointer: true, + showTicks: true, + ); + + expect(radialGaugeModel.toJson(), radialGaugeJson); + }); + + testWidgets('Radial gauge widget test with pointer', (widgetTester) async { + FlutterError.onError = ignoreOverflowErrors; + + RadialGaugeModel radialGaugeModel = RadialGaugeModel( + ntConnection: ntConnection, + preferences: preferences, + topic: 'Test/Double Value', + dataType: 'double', + period: 0.100, + startAngle: -140.0, + endAngle: 140.0, + minValue: -1.0, + maxValue: 1.0, + numberOfLabels: 10, + wrapValue: false, + showPointer: true, + showTicks: true, + ); + + await widgetTester.pumpWidget( + MaterialApp( + home: Scaffold( + body: ChangeNotifierProvider.value( + value: radialGaugeModel, + child: const RadialGauge(), + ), + ), + ), + ); + + await widgetTester.pumpAndSettle(); + + expect(find.byType(SfRadialGauge), findsOneWidget); + expect(find.text('-0.50'), findsOneWidget); + expect(find.byType(NeedlePointer), findsOneWidget); + }); + + testWidgets('Radial gauge widget test no pointer', (widgetTester) async { + FlutterError.onError = ignoreOverflowErrors; + + RadialGaugeModel radialGaugeModel = RadialGaugeModel( + ntConnection: ntConnection, + preferences: preferences, + topic: 'Test/Double Value', + dataType: 'double', + period: 0.100, + startAngle: -140.0, + endAngle: 140.0, + minValue: -1.0, + maxValue: 1.0, + numberOfLabels: 10, + wrapValue: false, + showPointer: false, + showTicks: false, + ); + + await widgetTester.pumpWidget( + MaterialApp( + home: Scaffold( + body: ChangeNotifierProvider.value( + value: radialGaugeModel, + child: const RadialGauge(), + ), + ), + ), + ); + + await widgetTester.pumpAndSettle(); + + expect(find.byType(SfRadialGauge), findsOneWidget); + expect(find.text('-0.50'), findsOneWidget); + expect(find.byType(NeedlePointer), findsNothing); + }); +} diff --git a/test/widgets/nt_widgets/single-topic/single_color_view_test.dart b/test/widgets/nt_widgets/single-topic/single_color_view_test.dart new file mode 100644 index 00000000..e10b3ee8 --- /dev/null +++ b/test/widgets/nt_widgets/single-topic/single_color_view_test.dart @@ -0,0 +1,112 @@ +import 'package:flutter/material.dart'; + +import 'package:flutter_colorpicker/flutter_colorpicker.dart'; +import 'package:flutter_test/flutter_test.dart'; +import 'package:provider/provider.dart'; +import 'package:shared_preferences/shared_preferences.dart'; + +import 'package:elastic_dashboard/services/nt4_client.dart'; +import 'package:elastic_dashboard/services/nt_connection.dart'; +import 'package:elastic_dashboard/services/nt_widget_builder.dart'; +import 'package:elastic_dashboard/widgets/nt_widgets/nt_widget.dart'; +import 'package:elastic_dashboard/widgets/nt_widgets/single_topic/single_color_view.dart'; +import '../../../test_util.dart'; + +void main() { + TestWidgetsFlutterBinding.ensureInitialized(); + + final Map singleColorViewJson = { + 'topic': 'Test/String Value', + 'data_type': 'string', + 'period': 0.100, + }; + + Finder findColor(Color color) => find.byWidgetPredicate( + (widget) => + widget is Container && + widget.decoration is BoxDecoration && + (widget.decoration as BoxDecoration).color != null && + (widget.decoration as BoxDecoration).color!.value == color.value, + ); + + late SharedPreferences preferences; + late NTConnection ntConnection; + + setUp(() async { + SharedPreferences.setMockInitialValues({}); + preferences = await SharedPreferences.getInstance(); + + ntConnection = createMockOnlineNT4( + virtualTopics: [ + NT4Topic( + name: 'Test/String Value', + type: NT4TypeStr.kString, + properties: {}, + ), + ], + virtualValues: { + 'Test/String Value': Colors.red.toHexString( + includeHashSign: true, + enableAlpha: false, + ), + }, + ); + }); + + test('Single color view from json', () { + NTWidgetModel singleColorViewModel = NTWidgetBuilder.buildNTModelFromJson( + ntConnection, + preferences, + 'Single Color View', + singleColorViewJson, + ); + + expect(singleColorViewModel.type, 'Single Color View'); + expect(singleColorViewModel.runtimeType, NTWidgetModel); + expect( + singleColorViewModel.getAvailableDisplayTypes(), + unorderedEquals([ + 'Text Display', + 'Single Color View', + ])); + }); + + test('Single color view to json', () { + NTWidgetModel singleColorViewModel = NTWidgetModel.createDefault( + ntConnection: ntConnection, + preferences: preferences, + type: 'Single Color View', + topic: 'Test/String Value', + dataType: 'string', + period: 0.100, + ); + + expect(singleColorViewModel.toJson(), singleColorViewJson); + }); + + testWidgets('Single color view widget test', (widgetTester) async { + FlutterError.onError = ignoreOverflowErrors; + + NTWidgetModel singleColorViewModel = NTWidgetBuilder.buildNTModelFromJson( + ntConnection, + preferences, + 'Single Color View', + singleColorViewJson, + ); + + await widgetTester.pumpWidget( + MaterialApp( + home: Scaffold( + body: ChangeNotifierProvider.value( + value: singleColorViewModel, + child: const SingleColorView(), + ), + ), + ), + ); + + await widgetTester.pumpAndSettle(); + + expect(findColor(Colors.red), findsOneWidget); + }); +} diff --git a/test/widgets/nt_widgets/single-topic/text_display_test.dart b/test/widgets/nt_widgets/single-topic/text_display_test.dart new file mode 100644 index 00000000..06e87f6c --- /dev/null +++ b/test/widgets/nt_widgets/single-topic/text_display_test.dart @@ -0,0 +1,564 @@ +import 'package:flutter/material.dart'; + +import 'package:flutter_test/flutter_test.dart'; +import 'package:provider/provider.dart'; +import 'package:shared_preferences/shared_preferences.dart'; + +import 'package:elastic_dashboard/services/nt4_client.dart'; +import 'package:elastic_dashboard/services/nt_connection.dart'; +import 'package:elastic_dashboard/services/nt_widget_builder.dart'; +import 'package:elastic_dashboard/widgets/nt_widgets/nt_widget.dart'; +import 'package:elastic_dashboard/widgets/nt_widgets/single_topic/text_display.dart'; +import '../../../test_util.dart'; + +void main() { + TestWidgetsFlutterBinding.ensureInitialized(); + + final Map textDisplayJson = { + 'topic': 'Test/Display Value', + 'data_type': 'double', + 'period': 0.100, + 'show_submit_button': true, + }; + + late SharedPreferences preferences; + late NTConnection ntConnection; + + setUp(() async { + SharedPreferences.setMockInitialValues({}); + preferences = await SharedPreferences.getInstance(); + + ntConnection = createMockOnlineNT4( + virtualTopics: [ + NT4Topic( + name: 'Test/Display Value', + type: NT4TypeStr.kFloat64, + properties: {}, + ), + ], + virtualValues: { + 'Test/Display Value': 0.000001, + }, + ); + }); + + test('Text display from json', () { + NTWidgetModel textDisplayModel = NTWidgetBuilder.buildNTModelFromJson( + ntConnection, + preferences, + 'Text Display', + textDisplayJson, + ); + + expect(textDisplayModel.type, 'Text Display'); + expect(textDisplayModel.runtimeType, TextDisplayModel); + + if (textDisplayModel is! TextDisplayModel) { + return; + } + + expect(textDisplayModel.showSubmitButton, isTrue); + }); + + test('Text display from alias name', () { + NTWidgetModel textDisplayModel = NTWidgetBuilder.buildNTModelFromJson( + ntConnection, + preferences, + 'Text View', + textDisplayJson, + ); + + expect(textDisplayModel.type, 'Text Display'); + expect(textDisplayModel.runtimeType, TextDisplayModel); + + if (textDisplayModel is! TextDisplayModel) { + return; + } + + expect(textDisplayModel.showSubmitButton, isTrue); + }); + + test('Text display to json', () { + TextDisplayModel textDisplayModel = TextDisplayModel( + ntConnection: ntConnection, + preferences: preferences, + topic: 'Test/Display Value', + dataType: 'double', + period: 0.100, + showSubmitButton: true, + ); + + expect(textDisplayModel.toJson(), textDisplayJson); + }); + + testWidgets('Text display widget test (double)', (widgetTester) async { + FlutterError.onError = ignoreOverflowErrors; + + TextDisplayModel textDisplayModel = TextDisplayModel( + ntConnection: ntConnection, + preferences: preferences, + topic: 'Test/Display Value', + dataType: 'double', + period: 0.100, + showSubmitButton: true, + ); + + await widgetTester.pumpWidget( + MaterialApp( + home: Scaffold( + body: ChangeNotifierProvider.value( + value: textDisplayModel, + child: const TextDisplay(), + ), + ), + ), + ); + + await widgetTester.pumpAndSettle(); + + expect(find.text('0.000001'), findsOneWidget); + expect(find.byTooltip('Publish Data'), findsOneWidget); + expect(find.byIcon(Icons.exit_to_app), findsOneWidget); + expect(find.byType(TextField), findsOneWidget); + + await widgetTester.enterText(find.byType(TextField), '3.53'); + expect(ntConnection.getLastAnnouncedValue('Test/Display Value'), 0.000001); + + await widgetTester.tap(find.byIcon(Icons.exit_to_app)); + await widgetTester.pumpAndSettle(); + + expect(ntConnection.getLastAnnouncedValue('Test/Display Value'), 3.53); + }); + + testWidgets('Text display widget test (int)', (widgetTester) async { + FlutterError.onError = ignoreOverflowErrors; + + NTConnection intNTConnection; + + TextDisplayModel textDisplayModel = TextDisplayModel( + ntConnection: intNTConnection = createMockOnlineNT4( + virtualTopics: [ + NT4Topic( + name: 'Test/Display Value', + type: NT4TypeStr.kInt, + properties: {}, + ), + ], + virtualValues: { + 'Test/Display Value': 0, + }, + ), + preferences: preferences, + topic: 'Test/Display Value', + dataType: 'int', + period: 0.100, + showSubmitButton: true, + ); + + await widgetTester.pumpWidget( + MaterialApp( + home: Scaffold( + body: ChangeNotifierProvider.value( + value: textDisplayModel, + child: const TextDisplay(), + ), + ), + ), + ); + + await widgetTester.pumpAndSettle(); + + expect(find.text('0'), findsOneWidget); + expect(find.byTooltip('Publish Data'), findsOneWidget); + expect(find.byIcon(Icons.exit_to_app), findsOneWidget); + expect(find.byType(TextField), findsOneWidget); + + await widgetTester.enterText(find.byType(TextField), '3000'); + expect(intNTConnection.getLastAnnouncedValue('Test/Display Value'), 0); + + await widgetTester.tap(find.byIcon(Icons.exit_to_app)); + await widgetTester.pumpAndSettle(); + + expect(intNTConnection.getLastAnnouncedValue('Test/Display Value'), 3000); + }); + + testWidgets('Text display widget test (boolean)', (widgetTester) async { + FlutterError.onError = ignoreOverflowErrors; + + NTConnection boolNTConnection; + + TextDisplayModel textDisplayModel = TextDisplayModel( + ntConnection: boolNTConnection = createMockOnlineNT4( + virtualTopics: [ + NT4Topic( + name: 'Test/Display Value', + type: NT4TypeStr.kBool, + properties: {}, + ), + ], + virtualValues: { + 'Test/Display Value': false, + }, + ), + preferences: preferences, + topic: 'Test/Display Value', + dataType: 'boolean', + period: 0.100, + showSubmitButton: true, + ); + + await widgetTester.pumpWidget( + MaterialApp( + home: Scaffold( + body: ChangeNotifierProvider.value( + value: textDisplayModel, + child: const TextDisplay(), + ), + ), + ), + ); + + await widgetTester.pumpAndSettle(); + + expect(find.text('false'), findsOneWidget); + expect(find.byTooltip('Publish Data'), findsOneWidget); + expect(find.byIcon(Icons.exit_to_app), findsOneWidget); + expect(find.byType(TextField), findsOneWidget); + + await widgetTester.enterText(find.byType(TextField), 'true'); + expect( + boolNTConnection.getLastAnnouncedValue('Test/Display Value'), isFalse); + + await widgetTester.tap(find.byIcon(Icons.exit_to_app)); + await widgetTester.pumpAndSettle(); + + expect( + boolNTConnection.getLastAnnouncedValue('Test/Display Value'), isTrue); + }); + + testWidgets('Text display widget test (string)', (widgetTester) async { + FlutterError.onError = ignoreOverflowErrors; + + NTConnection stringNTConnection; + + TextDisplayModel textDisplayModel = TextDisplayModel( + ntConnection: stringNTConnection = createMockOnlineNT4( + virtualTopics: [ + NT4Topic( + name: 'Test/Display Value', + type: NT4TypeStr.kString, + properties: {}, + ), + ], + virtualValues: { + 'Test/Display Value': 'Hello', + }, + ), + preferences: preferences, + topic: 'Test/Display Value', + dataType: 'string', + period: 0.100, + showSubmitButton: true, + ); + + await widgetTester.pumpWidget( + MaterialApp( + home: Scaffold( + body: ChangeNotifierProvider.value( + value: textDisplayModel, + child: const TextDisplay(), + ), + ), + ), + ); + + await widgetTester.pumpAndSettle(); + + expect(find.text('Hello'), findsOneWidget); + expect(find.byTooltip('Publish Data'), findsOneWidget); + expect(find.byIcon(Icons.exit_to_app), findsOneWidget); + expect(find.byType(TextField), findsOneWidget); + + await widgetTester.enterText(find.byType(TextField), 'I Edited This Text'); + expect(stringNTConnection.getLastAnnouncedValue('Test/Display Value'), + 'Hello'); + + await widgetTester.tap(find.byIcon(Icons.exit_to_app)); + await widgetTester.pumpAndSettle(); + + expect(stringNTConnection.getLastAnnouncedValue('Test/Display Value'), + 'I Edited This Text'); + }); + + testWidgets('Text display widget test (int[])', (widgetTester) async { + FlutterError.onError = ignoreOverflowErrors; + + NTConnection intArrNTConnection; + + TextDisplayModel textDisplayModel = TextDisplayModel( + ntConnection: intArrNTConnection = createMockOnlineNT4( + virtualTopics: [ + NT4Topic( + name: 'Test/Display Value', + type: NT4TypeStr.kIntArr, + properties: {}, + ), + ], + virtualValues: { + 'Test/Display Value': [0, 0], + }, + ), + preferences: preferences, + topic: 'Test/Display Value', + dataType: 'int[]', + period: 0.100, + showSubmitButton: true, + ); + + await widgetTester.pumpWidget( + MaterialApp( + home: Scaffold( + body: ChangeNotifierProvider.value( + value: textDisplayModel, + child: const TextDisplay(), + ), + ), + ), + ); + + await widgetTester.pumpAndSettle(); + + expect(find.text('[0, 0]'), findsOneWidget); + expect(find.byTooltip('Publish Data'), findsOneWidget); + expect(find.byIcon(Icons.exit_to_app), findsOneWidget); + expect(find.byType(TextField), findsOneWidget); + + await widgetTester.enterText(find.byType(TextField), '[1, 2, 3]'); + expect( + intArrNTConnection.getLastAnnouncedValue('Test/Display Value'), [0, 0]); + + await widgetTester.tap(find.byIcon(Icons.exit_to_app)); + await widgetTester.pumpAndSettle(); + + expect(intArrNTConnection.getLastAnnouncedValue('Test/Display Value'), + [1, 2, 3]); + }); + + testWidgets('Text display widget test (boolean[])', (widgetTester) async { + FlutterError.onError = ignoreOverflowErrors; + + NTConnection boolArrNTConnection; + + TextDisplayModel textDisplayModel = TextDisplayModel( + ntConnection: boolArrNTConnection = createMockOnlineNT4( + virtualTopics: [ + NT4Topic( + name: 'Test/Display Value', + type: NT4TypeStr.kBoolArr, + properties: {}, + ), + ], + virtualValues: { + 'Test/Display Value': [false, true], + }, + ), + preferences: preferences, + topic: 'Test/Display Value', + dataType: 'boolean[]', + period: 0.100, + showSubmitButton: true, + ); + + await widgetTester.pumpWidget( + MaterialApp( + home: Scaffold( + body: ChangeNotifierProvider.value( + value: textDisplayModel, + child: const TextDisplay(), + ), + ), + ), + ); + + await widgetTester.pumpAndSettle(); + + expect(find.text('[false, true]'), findsOneWidget); + expect(find.byTooltip('Publish Data'), findsOneWidget); + expect(find.byIcon(Icons.exit_to_app), findsOneWidget); + expect(find.byType(TextField), findsOneWidget); + + await widgetTester.enterText(find.byType(TextField), '[true, false, true]'); + expect(boolArrNTConnection.getLastAnnouncedValue('Test/Display Value'), + [false, true]); + + await widgetTester.tap(find.byIcon(Icons.exit_to_app)); + await widgetTester.pumpAndSettle(); + + expect(boolArrNTConnection.getLastAnnouncedValue('Test/Display Value'), + [true, false, true]); + }); + + testWidgets('Text display widget test (double[])', (widgetTester) async { + FlutterError.onError = ignoreOverflowErrors; + + NTConnection doubleArrNTConnection; + + TextDisplayModel textDisplayModel = TextDisplayModel( + ntConnection: doubleArrNTConnection = createMockOnlineNT4( + virtualTopics: [ + NT4Topic( + name: 'Test/Display Value', + type: NT4TypeStr.kFloat64Arr, + properties: {}, + ), + ], + virtualValues: { + 'Test/Display Value': [0.0, 0.0], + }, + ), + preferences: preferences, + topic: 'Test/Display Value', + dataType: 'double[]', + period: 0.100, + showSubmitButton: true, + ); + + await widgetTester.pumpWidget( + MaterialApp( + home: Scaffold( + body: ChangeNotifierProvider.value( + value: textDisplayModel, + child: const TextDisplay(), + ), + ), + ), + ); + + await widgetTester.pumpAndSettle(); + + expect(find.text('[0.0, 0.0]'), findsOneWidget); + expect(find.byTooltip('Publish Data'), findsOneWidget); + expect(find.byIcon(Icons.exit_to_app), findsOneWidget); + expect(find.byType(TextField), findsOneWidget); + + await widgetTester.enterText(find.byType(TextField), '[1.0, 2.0, 3.0]'); + expect(doubleArrNTConnection.getLastAnnouncedValue('Test/Display Value'), + [0.0, 0.0]); + + await widgetTester.tap(find.byIcon(Icons.exit_to_app)); + await widgetTester.pumpAndSettle(); + + expect(doubleArrNTConnection.getLastAnnouncedValue('Test/Display Value'), + [1.0, 2.0, 3.0]); + }); + + testWidgets('Text display widget test (string[])', (widgetTester) async { + FlutterError.onError = ignoreOverflowErrors; + + NTConnection stringArrNTConnection; + + TextDisplayModel textDisplayModel = TextDisplayModel( + ntConnection: stringArrNTConnection = createMockOnlineNT4( + virtualTopics: [ + NT4Topic( + name: 'Test/Display Value', + type: NT4TypeStr.kStringArr, + properties: {}, + ), + ], + virtualValues: { + 'Test/Display Value': ['Hello', 'There'], + }, + ), + preferences: preferences, + topic: 'Test/Display Value', + dataType: 'string[]', + period: 0.100, + showSubmitButton: true, + ); + + await widgetTester.pumpWidget( + MaterialApp( + home: Scaffold( + body: ChangeNotifierProvider.value( + value: textDisplayModel, + child: const TextDisplay(), + ), + ), + ), + ); + + await widgetTester.pumpAndSettle(); + + expect(find.text('[Hello, There]'), findsOneWidget); + expect(find.byTooltip('Publish Data'), findsOneWidget); + expect(find.byIcon(Icons.exit_to_app), findsOneWidget); + expect(find.byType(TextField), findsOneWidget); + + await widgetTester.enterText( + find.byType(TextField), '["I", "am", "very", "tired"]'); + expect(stringArrNTConnection.getLastAnnouncedValue('Test/Display Value'), + ['Hello', 'There']); + + await widgetTester.tap(find.byIcon(Icons.exit_to_app)); + await widgetTester.pumpAndSettle(); + + expect(stringArrNTConnection.getLastAnnouncedValue('Test/Display Value'), + ['I', 'am', 'very', 'tired']); + }); + + testWidgets('Text display widget test no submit button', + (widgetTester) async { + FlutterError.onError = ignoreOverflowErrors; + + NTConnection stringNTConnection; + + TextDisplayModel textDisplayModel = TextDisplayModel( + ntConnection: stringNTConnection = createMockOnlineNT4( + virtualTopics: [ + NT4Topic( + name: 'Test/Display Value', + type: NT4TypeStr.kString, + properties: {}, + ), + ], + virtualValues: { + 'Test/Display Value': 'There isn\'t a submit button', + }, + ), + preferences: preferences, + topic: 'Test/Display Value', + dataType: 'string', + period: 0.100, + showSubmitButton: false, + ); + + await widgetTester.pumpWidget( + MaterialApp( + home: Scaffold( + body: ChangeNotifierProvider.value( + value: textDisplayModel, + child: const TextDisplay(), + ), + ), + ), + ); + + await widgetTester.pumpAndSettle(); + + expect(find.text('There isn\'t a submit button'), findsOneWidget); + expect(find.byTooltip('Publish Data'), findsNothing); + expect(find.byIcon(Icons.exit_to_app), findsNothing); + expect(find.byType(TextField), findsOneWidget); + + await widgetTester.enterText( + find.byType(TextField), 'I\'m submitting this without a button!'); + expect(stringNTConnection.getLastAnnouncedValue('Test/Display Value'), + 'There isn\'t a submit button'); + await widgetTester.testTextInput.receiveAction(TextInputAction.done); + await widgetTester.pumpAndSettle(); + + expect(stringNTConnection.getLastAnnouncedValue('Test/Display Value'), + 'I\'m submitting this without a button!'); + }); +} diff --git a/test/widgets/nt_widgets/single-topic/toggle_button_test.dart b/test/widgets/nt_widgets/single-topic/toggle_button_test.dart new file mode 100644 index 00000000..31335ffc --- /dev/null +++ b/test/widgets/nt_widgets/single-topic/toggle_button_test.dart @@ -0,0 +1,114 @@ +import 'package:flutter/material.dart'; + +import 'package:flutter_test/flutter_test.dart'; +import 'package:provider/provider.dart'; +import 'package:shared_preferences/shared_preferences.dart'; + +import 'package:elastic_dashboard/services/nt4_client.dart'; +import 'package:elastic_dashboard/services/nt_connection.dart'; +import 'package:elastic_dashboard/services/nt_widget_builder.dart'; +import 'package:elastic_dashboard/widgets/nt_widgets/nt_widget.dart'; +import 'package:elastic_dashboard/widgets/nt_widgets/single_topic/toggle_button.dart'; +import '../../../test_util.dart'; + +void main() { + TestWidgetsFlutterBinding.ensureInitialized(); + + final Map toggleButtonJson = { + 'topic': 'Test/Boolean Value', + 'data_type': 'boolean', + 'period': 0.100, + }; + + late SharedPreferences preferences; + late NTConnection ntConnection; + + setUp(() async { + SharedPreferences.setMockInitialValues({}); + preferences = await SharedPreferences.getInstance(); + + ntConnection = createMockOnlineNT4( + virtualTopics: [ + NT4Topic( + name: 'Test/Boolean Value', + type: NT4TypeStr.kBool, + properties: {}, + ), + ], + virtualValues: { + 'Test/Boolean Value': false, + }, + ); + }); + + test('Toggle button from json', () { + NTWidgetModel toggleButtonModel = NTWidgetBuilder.buildNTModelFromJson( + ntConnection, + preferences, + 'Toggle Button', + toggleButtonJson, + ); + + expect(toggleButtonModel.type, 'Toggle Button'); + expect(toggleButtonModel.runtimeType, NTWidgetModel); + expect( + toggleButtonModel.getAvailableDisplayTypes(), + unorderedEquals([ + 'Boolean Box', + 'Toggle Button', + 'Toggle Switch', + 'Text Display', + ])); + }); + + test('Toggle button to json', () { + NTWidgetModel toggleButtonModel = NTWidgetModel.createDefault( + ntConnection: ntConnection, + preferences: preferences, + type: 'Toggle Button', + topic: 'Test/Boolean Value', + dataType: 'boolean', + period: 0.100, + ); + + expect(toggleButtonModel.toJson(), toggleButtonJson); + }); + + testWidgets('Toggle button widget test', (widgetTester) async { + FlutterError.onError = ignoreOverflowErrors; + + NTWidgetModel toggleButtonModel = NTWidgetBuilder.buildNTModelFromJson( + ntConnection, + preferences, + 'Toggle Button', + toggleButtonJson, + ); + + await widgetTester.pumpWidget( + MaterialApp( + home: Scaffold( + body: ChangeNotifierProvider.value( + value: toggleButtonModel, + child: const ToggleButton(), + ), + ), + ), + ); + + await widgetTester.pumpAndSettle(); + + expect(find.text('Boolean Value'), findsOneWidget); + + await widgetTester.tap(find.text('Boolean Value')); + toggleButtonModel.refresh(); + await widgetTester.pumpAndSettle(); + + expect(ntConnection.getLastAnnouncedValue('Test/Boolean Value'), isTrue); + + await widgetTester.tap(find.text('Boolean Value')); + toggleButtonModel.refresh(); + await widgetTester.pumpAndSettle(); + + expect(ntConnection.getLastAnnouncedValue('Test/Boolean Value'), isFalse); + }); +} diff --git a/test/widgets/nt_widgets/single-topic/toggle_switch_test.dart b/test/widgets/nt_widgets/single-topic/toggle_switch_test.dart new file mode 100644 index 00000000..1cb143c9 --- /dev/null +++ b/test/widgets/nt_widgets/single-topic/toggle_switch_test.dart @@ -0,0 +1,114 @@ +import 'package:flutter/material.dart'; + +import 'package:flutter_test/flutter_test.dart'; +import 'package:provider/provider.dart'; +import 'package:shared_preferences/shared_preferences.dart'; + +import 'package:elastic_dashboard/services/nt4_client.dart'; +import 'package:elastic_dashboard/services/nt_connection.dart'; +import 'package:elastic_dashboard/services/nt_widget_builder.dart'; +import 'package:elastic_dashboard/widgets/nt_widgets/nt_widget.dart'; +import 'package:elastic_dashboard/widgets/nt_widgets/single_topic/toggle_switch.dart'; +import '../../../test_util.dart'; + +void main() { + TestWidgetsFlutterBinding.ensureInitialized(); + + final Map toggleSwitchJson = { + 'topic': 'Test/Boolean Value', + 'data_type': 'boolean', + 'period': 0.100, + }; + + late SharedPreferences preferences; + late NTConnection ntConnection; + + setUp(() async { + SharedPreferences.setMockInitialValues({}); + preferences = await SharedPreferences.getInstance(); + + ntConnection = createMockOnlineNT4( + virtualTopics: [ + NT4Topic( + name: 'Test/Boolean Value', + type: NT4TypeStr.kBool, + properties: {}, + ), + ], + virtualValues: { + 'Test/Boolean Value': false, + }, + ); + }); + + test('Toggle switch from json', () { + NTWidgetModel toggleSwitchModel = NTWidgetBuilder.buildNTModelFromJson( + ntConnection, + preferences, + 'Toggle Switch', + toggleSwitchJson, + ); + + expect(toggleSwitchModel.type, 'Toggle Switch'); + expect(toggleSwitchModel.runtimeType, NTWidgetModel); + expect( + toggleSwitchModel.getAvailableDisplayTypes(), + unorderedEquals([ + 'Boolean Box', + 'Toggle Button', + 'Toggle Switch', + 'Text Display', + ])); + }); + + test('Toggle switch to json', () { + NTWidgetModel toggleSwitchModel = NTWidgetModel.createDefault( + ntConnection: ntConnection, + preferences: preferences, + type: 'Toggle Switch', + topic: 'Test/Boolean Value', + dataType: 'boolean', + period: 0.100, + ); + + expect(toggleSwitchModel.toJson(), toggleSwitchJson); + }); + + testWidgets('Toggle switch widget test', (widgetTester) async { + FlutterError.onError = ignoreOverflowErrors; + + NTWidgetModel toggleSwitchModel = NTWidgetBuilder.buildNTModelFromJson( + ntConnection, + preferences, + 'Toggle Switch', + toggleSwitchJson, + ); + + await widgetTester.pumpWidget( + MaterialApp( + home: Scaffold( + body: ChangeNotifierProvider.value( + value: toggleSwitchModel, + child: const ToggleSwitch(), + ), + ), + ), + ); + + await widgetTester.pumpAndSettle(); + + expect(find.byType(Switch), findsOneWidget); + + await widgetTester.tap(find.byType(Switch)); + toggleSwitchModel.refresh(); + await widgetTester.pumpAndSettle(); + + expect(ntConnection.getLastAnnouncedValue('Test/Boolean Value'), isTrue); + + await widgetTester.tap(find.byType(Switch)); + toggleSwitchModel.refresh(); + await widgetTester.pumpAndSettle(); + + expect(ntConnection.getLastAnnouncedValue('Test/Boolean Value'), isFalse); + }); +} diff --git a/test/widgets/nt_widgets/single-topic/voltage_view_test.dart b/test/widgets/nt_widgets/single-topic/voltage_view_test.dart new file mode 100644 index 00000000..4415df35 --- /dev/null +++ b/test/widgets/nt_widgets/single-topic/voltage_view_test.dart @@ -0,0 +1,212 @@ +import 'package:flutter/material.dart'; + +import 'package:flutter_test/flutter_test.dart'; +import 'package:provider/provider.dart'; +import 'package:shared_preferences/shared_preferences.dart'; +import 'package:syncfusion_flutter_gauges/gauges.dart'; + +import 'package:elastic_dashboard/services/nt4_client.dart'; +import 'package:elastic_dashboard/services/nt_connection.dart'; +import 'package:elastic_dashboard/services/nt_widget_builder.dart'; +import 'package:elastic_dashboard/widgets/nt_widgets/nt_widget.dart'; +import 'package:elastic_dashboard/widgets/nt_widgets/single_topic/voltage_view.dart'; +import '../../../test_util.dart'; + +void main() { + TestWidgetsFlutterBinding.ensureInitialized(); + + final Map voltageViewJson = { + 'topic': 'Test/Double Value', + 'data_type': 'double', + 'period': 0.100, + 'min_value': 4.0, + 'max_value': 13.0, + 'inverted': false, + 'orientation': 'horizontal', + }; + + late SharedPreferences preferences; + late NTConnection ntConnection; + + setUp(() async { + SharedPreferences.setMockInitialValues({}); + preferences = await SharedPreferences.getInstance(); + + ntConnection = createMockOnlineNT4( + virtualTopics: [ + NT4Topic( + name: 'Test/Double Value', + type: NT4TypeStr.kFloat64, + properties: {}, + ), + ], + virtualValues: { + 'Test/Double Value': 12.0, + }, + ); + }); + + test('Voltage view from json', () { + NTWidgetModel voltageViewModel = NTWidgetBuilder.buildNTModelFromJson( + ntConnection, + preferences, + 'Voltage View', + voltageViewJson, + ); + + expect(voltageViewModel.type, 'Voltage View'); + expect(voltageViewModel.runtimeType, VoltageViewModel); + expect( + voltageViewModel.getAvailableDisplayTypes(), + unorderedEquals([ + 'Text Display', + 'Number Bar', + 'Number Slider', + 'Graph', + 'Voltage View', + 'Radial Gauge', + 'Match Time', + ])); + + if (voltageViewModel is! VoltageViewModel) { + return; + } + + expect(voltageViewModel.minValue, 4.0); + expect(voltageViewModel.maxValue, 13.0); + expect(voltageViewModel.divisions, isNull); + expect(voltageViewModel.inverted, isFalse); + expect(voltageViewModel.orientation, 'horizontal'); + }); + + test('Voltage view to json', () { + VoltageViewModel voltageViewModel = VoltageViewModel( + ntConnection: ntConnection, + preferences: preferences, + topic: 'Test/Double Value', + dataType: 'double', + period: 0.100, + minValue: 4.0, + maxValue: 13.0, + divisions: null, + inverted: false, + orientation: 'horizontal', + ); + + expect(voltageViewModel.toJson(), voltageViewJson); + }); + + testWidgets('Voltage view widget test horizontal', (widgetTester) async { + FlutterError.onError = ignoreOverflowErrors; + + NTWidgetModel voltageViewModel = VoltageViewModel( + ntConnection: ntConnection, + preferences: preferences, + topic: 'Test/Double Value', + dataType: 'double', + period: 0.100, + minValue: -5.0, + maxValue: 5.0, + divisions: null, + inverted: false, + orientation: 'horizontal', + ); + + await widgetTester.pumpWidget( + MaterialApp( + home: Scaffold( + body: ChangeNotifierProvider.value( + value: voltageViewModel, + child: const VoltageView(), + ), + ), + ), + ); + + await widgetTester.pumpAndSettle(); + + expect(find.text('12.00 V'), findsOneWidget); + expect(find.byType(SfLinearGauge), findsOneWidget); + + expect( + (find.byType(SfLinearGauge).evaluate().first.widget as SfLinearGauge) + .orientation, + LinearGaugeOrientation.horizontal); + }); + + testWidgets('Voltage view widget test vertical', (widgetTester) async { + FlutterError.onError = ignoreOverflowErrors; + + NTWidgetModel voltageViewModel = VoltageViewModel( + ntConnection: ntConnection, + preferences: preferences, + topic: 'Test/Double Value', + dataType: 'double', + period: 0.100, + minValue: 4.0, + maxValue: 13.0, + divisions: null, + inverted: false, + orientation: 'vertical', + ); + + await widgetTester.pumpWidget( + MaterialApp( + home: Scaffold( + body: ChangeNotifierProvider.value( + value: voltageViewModel, + child: const VoltageView(), + ), + ), + ), + ); + + await widgetTester.pumpAndSettle(); + + expect(find.text('12.00 V'), findsOneWidget); + expect(find.byType(SfLinearGauge), findsOneWidget); + + expect( + (find.byType(SfLinearGauge).evaluate().first.widget as SfLinearGauge) + .orientation, + LinearGaugeOrientation.vertical); + }); + + testWidgets('Voltage view widget test with divisions', (widgetTester) async { + FlutterError.onError = ignoreOverflowErrors; + + NTWidgetModel voltageViewModel = VoltageViewModel( + ntConnection: ntConnection, + preferences: preferences, + topic: 'Test/Double Value', + dataType: 'double', + period: 0.100, + minValue: 4.0, + maxValue: 13.0, + divisions: 11, + inverted: false, + orientation: 'horizontal', + ); + + await widgetTester.pumpWidget( + MaterialApp( + home: Scaffold( + body: ChangeNotifierProvider.value( + value: voltageViewModel, + child: const VoltageView(), + ), + ), + ), + ); + + await widgetTester.pumpAndSettle(); + + expect(find.text('12.00 V'), findsOneWidget); + expect(find.byType(SfLinearGauge), findsOneWidget); + + expect( + (find.byType(SfLinearGauge).evaluate().first.widget as SfLinearGauge) + .interval, + 0.9); + }); +} diff --git a/test/widgets/settings_dialog_test.dart b/test/widgets/settings_dialog_test.dart index d231f0f5..e21be1cc 100644 --- a/test/widgets/settings_dialog_test.dart +++ b/test/widgets/settings_dialog_test.dart @@ -1,5 +1,6 @@ import 'package:flutter/material.dart'; +import 'package:flex_seed_scheme/flex_seed_scheme.dart'; import 'package:flutter_colorpicker/flutter_colorpicker.dart'; import 'package:flutter_test/flutter_test.dart'; import 'package:mockito/annotations.dart'; @@ -41,6 +42,8 @@ class FakeSettingsMethods { void changeDefaultPeriod() {} void changeDefaultGraphPeriod() {} + + void changeThemeVariant() {} } void main() { @@ -63,6 +66,7 @@ void main() { PrefKeys.layoutLocked: false, PrefKeys.defaultPeriod: 0.10, PrefKeys.defaultGraphPeriod: 0.033, + PrefKeys.themeVariant: FlexSchemeVariant.chroma.variantName, }); preferences = await SharedPreferences.getInstance(); @@ -76,11 +80,11 @@ void main() { testWidgets('Settings Dialog', (widgetTester) async { FlutterError.onError = ignoreOverflowErrors; - setupMockOfflineNT4(); await widgetTester.pumpWidget(MaterialApp( home: Scaffold( body: SettingsDialog( + ntConnection: createMockOfflineNT4(), preferences: preferences, ), ), @@ -101,6 +105,7 @@ void main() { expect(find.text('Lock Layout'), findsOneWidget); expect(find.text('Default Period'), findsOneWidget); expect(find.text('Default Graph Period'), findsOneWidget); + expect(find.text('Theme Variant'), findsOneWidget); final closeButton = find.widgetWithText(TextButton, 'Close'); @@ -112,17 +117,17 @@ void main() { testWidgets('Change team number', (widgetTester) async { FlutterError.onError = ignoreOverflowErrors; - setupMockOfflineNT4(); await widgetTester.pumpWidget(MaterialApp( home: Scaffold( body: SettingsDialog( + ntConnection: createMockOfflineNT4(), + preferences: preferences, onTeamNumberChanged: (data) async { fakeSettings.changeTeamNumber(); await preferences.setInt(PrefKeys.teamNumber, int.parse(data!)); }, - preferences: preferences, ), ), )); @@ -146,17 +151,17 @@ void main() { testWidgets('Change team color', (widgetTester) async { FlutterError.onError = ignoreOverflowErrors; - setupMockOfflineNT4(); await widgetTester.pumpWidget(MaterialApp( home: Scaffold( body: SettingsDialog( + ntConnection: createMockOnlineNT4(), + preferences: preferences, onColorChanged: (color) async { fakeSettings.changeColor(); await preferences.setInt(PrefKeys.teamColor, color.value); }, - preferences: preferences, ), ), )); @@ -200,19 +205,72 @@ void main() { verify(fakeSettings.changeColor()).called(greaterThanOrEqualTo(1)); }); + testWidgets('Change theme variant', (widgetTester) async { + FlutterError.onError = ignoreOverflowErrors; + + await widgetTester.pumpWidget(MaterialApp( + home: Scaffold( + body: SettingsDialog( + ntConnection: createMockOfflineNT4(), + onThemeVariantChanged: (variant) async { + fakeSettings.changeThemeVariant(); + + await preferences.setString( + PrefKeys.themeVariant, variant.variantName); + }, + preferences: preferences, + ), + ), + )); + + await widgetTester.pumpAndSettle(); + + final themeVariantDropdown = + find.widgetWithText(DialogDropdownChooser, 'Chroma'); + + expect(themeVariantDropdown, findsOneWidget); + + await widgetTester.tap(themeVariantDropdown); + await widgetTester.pumpAndSettle(); + + expect(find.text('Chroma'), findsNWidgets(2)); + expect(find.text('Material-3 Legacy (Default)'), findsOneWidget); + expect(find.text('Material-3 Legacy'), findsNothing); + + await widgetTester.tap(find.text('Material-3 Legacy (Default)')); + await widgetTester.pumpAndSettle(); + + expect(preferences.getString(PrefKeys.themeVariant), + FlexSchemeVariant.material3Legacy.variantName); + + verify(fakeSettings.changeThemeVariant()).called(1); + + final newThemeVariantDropdown = find.widgetWithText( + DialogDropdownChooser, 'Material-3 Legacy (Default)'); + + expect(newThemeVariantDropdown, findsOneWidget); + + // Now the safety mecahnism to add unknown variants should activate + await widgetTester.tap(newThemeVariantDropdown); + await widgetTester.pumpAndSettle(); + + expect(find.text('Material-3 Legacy (Default)'), findsNWidgets(2)); + expect(find.text('Material-3 Legacy'), findsOneWidget); + }); + testWidgets('Toggle grid', (widgetTester) async { FlutterError.onError = ignoreOverflowErrors; - setupMockOfflineNT4(); await widgetTester.pumpWidget(MaterialApp( home: Scaffold( body: SettingsDialog( + ntConnection: createMockOfflineNT4(), + preferences: preferences, onGridToggle: (value) async { fakeSettings.changeShowGrid(); await preferences.setBool(PrefKeys.showGrid, value); }, - preferences: preferences, ), ), )); @@ -244,17 +302,17 @@ void main() { testWidgets('Change grid size', (widgetTester) async { FlutterError.onError = ignoreOverflowErrors; - setupMockOfflineNT4(); await widgetTester.pumpWidget(MaterialApp( home: Scaffold( body: SettingsDialog( + ntConnection: createMockOfflineNT4(), + preferences: preferences, onGridSizeChanged: (gridSize) async { fakeSettings.changeGridSize(); await preferences.setInt(PrefKeys.gridSize, int.parse(gridSize!)); }, - preferences: preferences, ), ), )); @@ -275,18 +333,19 @@ void main() { testWidgets('Change corner radius', (widgetTester) async { FlutterError.onError = ignoreOverflowErrors; - setupMockOfflineNT4(); + createMockOfflineNT4(); await widgetTester.pumpWidget(MaterialApp( home: Scaffold( body: SettingsDialog( + ntConnection: createMockOfflineNT4(), + preferences: preferences, onCornerRadiusChanged: (radius) async { fakeSettings.changeCornerRadius(); await preferences.setDouble( PrefKeys.cornerRadius, double.parse(radius!)); }, - preferences: preferences, ), ), )); @@ -308,17 +367,17 @@ void main() { testWidgets('Toggle driver station auto resize', (widgetTester) async { FlutterError.onError = ignoreOverflowErrors; - setupMockOfflineNT4(); await widgetTester.pumpWidget(MaterialApp( home: Scaffold( body: SettingsDialog( + ntConnection: createMockOfflineNT4(), + preferences: preferences, onResizeToDSChanged: (value) async { fakeSettings.changeDSAutoResize(); await preferences.setBool(PrefKeys.autoResizeToDS, value); }, - preferences: preferences, ), ), )); @@ -351,17 +410,17 @@ void main() { testWidgets('Toggle remember window position', (widgetTester) async { FlutterError.onError = ignoreOverflowErrors; - setupMockOfflineNT4(); await widgetTester.pumpWidget(MaterialApp( home: Scaffold( body: SettingsDialog( + ntConnection: createMockOfflineNT4(), + preferences: preferences, onRememberWindowPositionChanged: (value) async { fakeSettings.changeRememberWindow(); await preferences.setBool(PrefKeys.rememberWindowPosition, value); }, - preferences: preferences, ), ), )); @@ -394,17 +453,17 @@ void main() { testWidgets('Toggle lock layout', (widgetTester) async { FlutterError.onError = ignoreOverflowErrors; - setupMockOfflineNT4(); await widgetTester.pumpWidget(MaterialApp( home: Scaffold( body: SettingsDialog( + preferences: preferences, + ntConnection: createMockOfflineNT4(), onLayoutLock: (value) async { fakeSettings.changeLockLayout(); await preferences.setBool(PrefKeys.layoutLocked, value); }, - preferences: preferences, ), ), )); @@ -437,11 +496,11 @@ void main() { testWidgets('Change IP address mode', (widgetTester) async { FlutterError.onError = ignoreOverflowErrors; - setupMockOfflineNT4(); await widgetTester.pumpWidget(MaterialApp( home: Scaffold( body: SettingsDialog( + ntConnection: createMockOfflineNT4(), preferences: preferences, onIPAddressModeChanged: (mode) { fakeSettings.changeIPAddressMode(); @@ -482,17 +541,17 @@ void main() { testWidgets('Change IP address', (widgetTester) async { FlutterError.onError = ignoreOverflowErrors; - setupMockOfflineNT4(); await widgetTester.pumpWidget(MaterialApp( home: Scaffold( body: SettingsDialog( + ntConnection: createMockOfflineNT4(), + preferences: preferences, onIPAddressChanged: (data) async { fakeSettings.changeIPAddress(); await preferences.setString(PrefKeys.ipAddress, data!); }, - preferences: preferences, ), ), )); @@ -513,18 +572,18 @@ void main() { testWidgets('Change default period', (widgetTester) async { FlutterError.onError = ignoreOverflowErrors; - setupMockOfflineNT4(); await widgetTester.pumpWidget(MaterialApp( home: Scaffold( body: SettingsDialog( + ntConnection: createMockOfflineNT4(), + preferences: preferences, onDefaultPeriodChanged: (period) async { fakeSettings.changeDefaultPeriod(); await preferences.setDouble( PrefKeys.defaultPeriod, double.parse(period!)); }, - preferences: preferences, ), ), )); @@ -545,18 +604,18 @@ void main() { testWidgets('Change default graph period', (widgetTester) async { FlutterError.onError = ignoreOverflowErrors; - setupMockOfflineNT4(); await widgetTester.pumpWidget(MaterialApp( home: Scaffold( body: SettingsDialog( + ntConnection: createMockOfflineNT4(), + preferences: preferences, onDefaultGraphPeriodChanged: (period) async { fakeSettings.changeDefaultGraphPeriod(); await preferences.setDouble( PrefKeys.defaultGraphPeriod, double.parse(period!)); }, - preferences: preferences, ), ), )); diff --git a/test/widgets/tab_grid_test.dart b/test/widgets/tab_grid_test.dart index 642cf823..5d3544df 100644 --- a/test/widgets/tab_grid_test.dart +++ b/test/widgets/tab_grid_test.dart @@ -6,6 +6,7 @@ import 'package:flutter/material.dart'; import 'package:flutter_test/flutter_test.dart'; import 'package:provider/provider.dart'; +import 'package:shared_preferences/shared_preferences.dart'; import 'package:elastic_dashboard/services/field_images.dart'; import 'package:elastic_dashboard/widgets/dialog_widgets/dialog_text_input.dart'; @@ -45,9 +46,9 @@ void main() async { late String jsonString; late Map jsonData; + late SharedPreferences preferences; setUpAll(() async { - setupMockOfflineNT4(); await FieldImages.loadFields('assets/fields/'); String filePath = @@ -55,6 +56,9 @@ void main() async { jsonString = File(filePath).readAsStringSync(); jsonData = jsonDecode(jsonString); + + SharedPreferences.setMockInitialValues({}); + preferences = await SharedPreferences.getInstance(); }); testWidgets('Tab grid loading (Tab 1)', (widgetTester) async { @@ -73,12 +77,14 @@ void main() async { await widgetTester.pumpWidget( MaterialApp( home: Scaffold( - body: ChangeNotifierProvider( - create: (context) => TabGridModel(), - child: TabGrid.fromJson( + body: ChangeNotifierProvider.value( + value: TabGridModel.fromJson( + ntConnection: createMockOfflineNT4(), + preferences: preferences, jsonData: jsonData['tabs'][0]['grid_layout'], onAddWidgetPressed: () {}, ), + child: const TabGrid(), ), ), ), @@ -120,12 +126,14 @@ void main() async { await widgetTester.pumpWidget( MaterialApp( home: Scaffold( - body: ChangeNotifierProvider( - create: (context) => TabGridModel(), - child: TabGrid.fromJson( + body: ChangeNotifierProvider.value( + value: TabGridModel.fromJson( + ntConnection: createMockOfflineNT4(), + preferences: preferences, jsonData: jsonData['tabs'][1]['grid_layout'], onAddWidgetPressed: () {}, ), + child: const TabGrid(), ), ), ), @@ -160,13 +168,14 @@ void main() async { await widgetTester.pumpWidget( MaterialApp( home: Scaffold( - body: ChangeNotifierProvider( - create: (context) => TabGridModel(), - child: TabGrid.fromJson( - key: GlobalKey(), + body: ChangeNotifierProvider.value( + value: TabGridModel.fromJson( + ntConnection: createMockOfflineNT4(), + preferences: preferences, jsonData: jsonData['tabs'][0]['grid_layout'], onAddWidgetPressed: () {}, ), + child: const TabGrid(), ), ), ), @@ -236,13 +245,14 @@ void main() async { await widgetTester.pumpWidget( MaterialApp( home: Scaffold( - body: ChangeNotifierProvider( - create: (context) => TabGridModel(), - child: TabGrid.fromJson( - key: GlobalKey(), + body: ChangeNotifierProvider.value( + value: TabGridModel.fromJson( + ntConnection: createMockOfflineNT4(), + preferences: preferences, jsonData: jsonData['tabs'][0]['grid_layout'], onAddWidgetPressed: () {}, ), + child: const TabGrid(), ), ), ), @@ -295,13 +305,14 @@ void main() async { await widgetTester.pumpWidget( MaterialApp( home: Scaffold( - body: ChangeNotifierProvider( - create: (context) => TabGridModel(), - child: TabGrid.fromJson( - key: GlobalKey(), + body: ChangeNotifierProvider.value( + value: TabGridModel.fromJson( + ntConnection: createMockOfflineNT4(), + preferences: preferences, jsonData: jsonData['tabs'][0]['grid_layout'], onAddWidgetPressed: () {}, ), + child: const TabGrid(), ), ), ), diff --git a/test_resources/test-layout.json b/test_resources/test-layout.json index 0cfaae7f..951427b9 100644 --- a/test_resources/test-layout.json +++ b/test_resources/test-layout.json @@ -362,7 +362,7 @@ "min_value": -1.0, "max_value": 1.0, "divisions": 5, - "publish_all": false + "update_continuously": false } }, { @@ -376,8 +376,6 @@ "topic": "/SmartDashboard/Voltage", "period": 0.033, "time_displayed": 5.0, - "min_value": null, - "max_value": null, "color": 4278238420, "line_width": 2.0 } From b76caa181235aa70c0ecd3cd498b021767fd9da5 Mon Sep 17 00:00:00 2001 From: DanPeled <98838880+DanPeled@users.noreply.github.com> Date: Sat, 20 Jul 2024 19:19:21 +0300 Subject: [PATCH 4/9] moved angle offset to match with new changes merged from original repo, added type hints to getters --- .../multi-topic/yagsl_swerve_drive.dart | 157 +++++++++++------- 1 file changed, 97 insertions(+), 60 deletions(-) diff --git a/lib/widgets/nt_widgets/multi-topic/yagsl_swerve_drive.dart b/lib/widgets/nt_widgets/multi-topic/yagsl_swerve_drive.dart index 2d75e283..ea367342 100644 --- a/lib/widgets/nt_widgets/multi-topic/yagsl_swerve_drive.dart +++ b/lib/widgets/nt_widgets/multi-topic/yagsl_swerve_drive.dart @@ -1,10 +1,13 @@ import 'dart:math'; + import 'package:elastic_dashboard/services/text_formatter_builder.dart'; import 'package:elastic_dashboard/widgets/dialog_widgets/dialog_text_input.dart'; import 'package:flutter/material.dart'; + import 'package:dot_cast/dot_cast.dart'; import 'package:provider/provider.dart'; import 'package:vector_math/vector_math_64.dart' show radians; + import 'package:elastic_dashboard/widgets/dialog_widgets/dialog_toggle_switch.dart'; import 'package:elastic_dashboard/widgets/nt_widgets/nt_widget.dart'; @@ -12,9 +15,39 @@ class YAGSLSwerveDriveModel extends NTWidgetModel { @override String type = YAGSLSwerveDrive.widgetType; - double _angleOffset = 0.0; - bool _showRobotRotation; - bool _showDesiredStates; + String get measuredStatesTopic => '$topic/measuredStates'; + String get desiredStatesTopic => '$topic/desiredStates'; + String get robotRotationTopic => '$topic/robotRotation'; + String get maxSpeedTopic => '$topic/maxSpeed'; + String get robotWidthTopic => '$topic/sizeLeftRight'; + String get robotLengthTopic => '$topic/sizeFrontBack'; + String get rotationUnitTopic => '$topic/rotationUnit'; + + bool _showRobotRotation = true; + bool _showDesiredStates = true; + double _angleOffset = + 0; // Modifiable angle offset to allow all kinds of swerve libraries setups + + bool get showRobotRotation => _showRobotRotation; + + set showRobotRotation(value) { + _showRobotRotation = value; + refresh(); + } + + bool get showDesiredStates => _showDesiredStates; + + set showDesiredStates(value) { + _showDesiredStates = value; + refresh(); + } + + double get angleOffset => _angleOffset; + + set angleOffset(value) { + _angleOffset = value; + refresh(); + } YAGSLSwerveDriveModel({ required super.ntConnection, @@ -35,6 +68,7 @@ class YAGSLSwerveDriveModel extends NTWidgetModel { }) : super.fromJson(jsonData: jsonData) { _showRobotRotation = tryCast(jsonData['show_robot_rotation']) ?? true; _showDesiredStates = tryCast(jsonData['show_desired_states']) ?? true; + _angleOffset = tryCast(jsonData['angle_offset']) ?? 0.0; } @override @@ -49,9 +83,9 @@ class YAGSLSwerveDriveModel extends NTWidgetModel { @override List getEditProperties(BuildContext context) { - String rotationUnit = + String angleUnit = tryCast(ntConnection.getLastAnnouncedValue(rotationUnitTopic)) ?? - 'radians'; + "radians"; return [ Row( children: [ @@ -59,14 +93,18 @@ class YAGSLSwerveDriveModel extends NTWidgetModel { child: DialogToggleSwitch( initialValue: _showRobotRotation, label: 'Show Robot Rotation', - onToggle: (value) => showRobotRotation = value, + onToggle: (value) { + showRobotRotation = value; + }, ), ), Flexible( child: DialogToggleSwitch( initialValue: _showDesiredStates, label: 'Show Desired States', - onToggle: (value) => showDesiredStates = value, + onToggle: (value) { + showDesiredStates = value; + }, ), ), ], @@ -76,15 +114,12 @@ class YAGSLSwerveDriveModel extends NTWidgetModel { children: [ Flexible( child: DialogTextInput( - onSubmit: (value) { - double? newOffset = double.tryParse(value); - if (newOffset != null) { - _angleOffset = newOffset; - } + initialText: _angleOffset.toString(), + label: 'Angle Offset ($angleUnit)', + onSubmit: (String value) async { + angleOffset = double.parse(value); }, formatter: TextFormatterBuilder.decimalTextFormatter(), - label: 'Angle Offset ($rotationUnit)', - initialText: _angleOffset.toString(), ), ), ], @@ -94,27 +129,27 @@ class YAGSLSwerveDriveModel extends NTWidgetModel { @override List getCurrentData() { - List rawMeasuredStates = + List measuredStatesRaw = tryCast(ntConnection.getLastAnnouncedValue(measuredStatesTopic)) ?? []; - List rawDesiredStates = + List desiredStatesRaw = tryCast(ntConnection.getLastAnnouncedValue(desiredStatesTopic)) ?? []; - // Filter and cast to List List measuredStates = - List.from(rawMeasuredStates.whereType()); - List desiredStates = - List.from(rawDesiredStates.whereType()); + measuredStatesRaw.whereType().toList(); + List desiredStates = desiredStatesRaw.whereType().toList(); double width = tryCast(ntConnection.getLastAnnouncedValue(robotWidthTopic)) ?? 1.0; double length = tryCast(ntConnection.getLastAnnouncedValue(robotLengthTopic)) ?? width; + String rotationUnit = tryCast(ntConnection.getLastAnnouncedValue(rotationUnitTopic)) ?? 'radians'; + double robotAngle = tryCast(ntConnection.getLastAnnouncedValue(robotRotationTopic)) ?? 0.0; - robotAngle += _angleOffset; + double maxSpeed = tryCast(ntConnection.getLastAnnouncedValue(maxSpeedTopic)) ?? 4.5; @@ -149,11 +184,10 @@ class YAGSLSwerveDrive extends NTWidget { .getLastAnnouncedValue(model.desiredStatesTopic)) ?? []; - // Filter and cast to List - List measuredStates = - List.from(rawMeasuredStates.whereType()); - List desiredStates = - List.from(rawDesiredStates.whereType()); + List measuredStates = + measuredStatesRaw.whereType().toList(); + List desiredStates = + desiredStatesRaw.whereType().toList(); double width = tryCast(model.ntConnection .getLastAnnouncedValue(model.robotWidthTopic)) ?? @@ -162,11 +196,15 @@ class YAGSLSwerveDrive extends NTWidget { .getLastAnnouncedValue(model.robotLengthTopic)) ?? width; - width = width > 0.0 ? width : 1.0; - length = length > 0.0 ? length : 0.0; + if (width <= 0.0) { + width = 1.0; + } + if (length <= 0.0) { + length = 0.0; + } - double sizeRatio = min(length, width) / max(length, width); - double lengthWidthRatio = length / width; + double sizeRatio = min(length, width) / max(length, width); + double lengthWidthRatio = length / width; String rotationUnit = tryCast(model.ntConnection .getLastAnnouncedValue(model.rotationUnitTopic)) ?? @@ -174,13 +212,13 @@ class YAGSLSwerveDrive extends NTWidget { double robotAngle = tryCast(model.ntConnection .getLastAnnouncedValue(model.robotRotationTopic)) ?? - 0.0; + 0.0 + model.angleOffset; - if (rotationUnit == 'degrees') { - robotAngle = radians(robotAngle + model._angleOffset); - } else if (rotationUnit == 'rotations') { - robotAngle *= 2 * pi + model._angleOffset; - } + if (rotationUnit == 'degrees') { + robotAngle = radians(robotAngle); + } else if (rotationUnit == 'rotations') { + robotAngle *= 2 * pi; + } double maxSpeed = tryCast(model.ntConnection .getLastAnnouncedValue(model.maxSpeedTopic)) ?? @@ -190,32 +228,31 @@ class YAGSLSwerveDrive extends NTWidget { maxSpeed = 4.5; } - return LayoutBuilder( - builder: (context, constraints) { - double maxSideLength = - min(constraints.maxWidth, constraints.maxHeight) * - 0.9 * - sizeRatio; - return Transform.rotate( - angle: model.showRobotRotation ? -robotAngle : 0.0, - child: SizedBox( - width: maxSideLength / lengthWidthRatio, - height: maxSideLength * lengthWidthRatio, - child: CustomPaint( - painter: SwerveDrivePainter( - rotationUnit: rotationUnit, - maxSpeed: maxSpeed, - moduleStates: measuredStates, - desiredStates: - model.showDesiredStates ? desiredStates : [], - ), + return LayoutBuilder( + builder: (context, constraints) { + double maxSideLength = + min(constraints.maxWidth, constraints.maxHeight) * + 0.9 * + sizeRatio; + return Transform.rotate( + angle: (model.showRobotRotation) ? -robotAngle : 0.0, + child: SizedBox( + width: maxSideLength / lengthWidthRatio, + height: maxSideLength * lengthWidthRatio, + child: CustomPaint( + painter: SwerveDrivePainter( + rotationUnit: rotationUnit, + maxSpeed: maxSpeed, + moduleStates: measuredStates, + desiredStates: + (model.showDesiredStates) ? desiredStates : [], ), ), - ); - }, - ); - }, - ), + ), + ); + }, + ); + }, ); } } From 2d9e1c1123b61cdcfb3c39f3a1b8b1f609b48b4d Mon Sep 17 00:00:00 2001 From: DanPeled <98838880+DanPeled@users.noreply.github.com> Date: Fri, 2 Aug 2024 18:00:17 +0300 Subject: [PATCH 5/9] added tests --- .../multi-topic/yagsl_swerve_drive.dart | 6 ++++-- .../multi-topic/yagsl_swerve_drive_test.dart | 16 +++++++++------- 2 files changed, 13 insertions(+), 9 deletions(-) diff --git a/lib/widgets/nt_widgets/multi-topic/yagsl_swerve_drive.dart b/lib/widgets/nt_widgets/multi-topic/yagsl_swerve_drive.dart index ea367342..b99d3ee6 100644 --- a/lib/widgets/nt_widgets/multi-topic/yagsl_swerve_drive.dart +++ b/lib/widgets/nt_widgets/multi-topic/yagsl_swerve_drive.dart @@ -1,13 +1,13 @@ import 'dart:math'; -import 'package:elastic_dashboard/services/text_formatter_builder.dart'; -import 'package:elastic_dashboard/widgets/dialog_widgets/dialog_text_input.dart'; import 'package:flutter/material.dart'; import 'package:dot_cast/dot_cast.dart'; import 'package:provider/provider.dart'; import 'package:vector_math/vector_math_64.dart' show radians; +import 'package:elastic_dashboard/services/text_formatter_builder.dart'; +import 'package:elastic_dashboard/widgets/dialog_widgets/dialog_text_input.dart'; import 'package:elastic_dashboard/widgets/dialog_widgets/dialog_toggle_switch.dart'; import 'package:elastic_dashboard/widgets/nt_widgets/nt_widget.dart'; @@ -55,10 +55,12 @@ class YAGSLSwerveDriveModel extends NTWidgetModel { required super.topic, bool showRobotRotation = true, bool showDesiredStates = true, + double angleOffset = 0.0, super.dataType, super.period, }) : _showDesiredStates = showDesiredStates, _showRobotRotation = showRobotRotation, + _angleOffset = angleOffset, super(); YAGSLSwerveDriveModel.fromJson({ diff --git a/test/widgets/nt_widgets/multi-topic/yagsl_swerve_drive_test.dart b/test/widgets/nt_widgets/multi-topic/yagsl_swerve_drive_test.dart index 8d770c1e..ac23b06c 100644 --- a/test/widgets/nt_widgets/multi-topic/yagsl_swerve_drive_test.dart +++ b/test/widgets/nt_widgets/multi-topic/yagsl_swerve_drive_test.dart @@ -18,6 +18,7 @@ void main() { 'period': 0.100, 'show_robot_rotation': true, 'show_desired_states': true, + 'angle_offset': 90.0, }; late SharedPreferences preferences; @@ -47,17 +48,18 @@ void main() { expect(yagslSwerveModel.showRobotRotation, isTrue); expect(yagslSwerveModel.showDesiredStates, isTrue); + expect(yagslSwerveModel.angleOffset, 90.0); }); test('YAGSL swerve drive to json', () { YAGSLSwerveDriveModel yagslSwerveModel = YAGSLSwerveDriveModel( - ntConnection: ntConnection, - preferences: preferences, - topic: 'Test/YAGSL Swerve Drive', - period: 0.100, - showRobotRotation: true, - showDesiredStates: true, - ); + ntConnection: ntConnection, + preferences: preferences, + topic: 'Test/YAGSL Swerve Drive', + period: 0.100, + showRobotRotation: true, + showDesiredStates: true, + angleOffset: 90.0); expect(yagslSwerveModel.toJson(), yagslSwerveJson); }); From abc3db31a0012963a119531867b66130d951ed54 Mon Sep 17 00:00:00 2001 From: DanPeled <98838880+DanPeled@users.noreply.github.com> Date: Mon, 19 Aug 2024 20:31:37 +0300 Subject: [PATCH 6/9] fixed offset converstion to be automatic (always be entered as deg and automatically converted to rad if needed), replaced double.parse to double.tryParse --- .../multi-topic/yagsl_swerve_drive.dart | 21 ++++++++++++------- 1 file changed, 14 insertions(+), 7 deletions(-) diff --git a/lib/widgets/nt_widgets/multi-topic/yagsl_swerve_drive.dart b/lib/widgets/nt_widgets/multi-topic/yagsl_swerve_drive.dart index b99d3ee6..ed8f7fe4 100644 --- a/lib/widgets/nt_widgets/multi-topic/yagsl_swerve_drive.dart +++ b/lib/widgets/nt_widgets/multi-topic/yagsl_swerve_drive.dart @@ -44,7 +44,7 @@ class YAGSLSwerveDriveModel extends NTWidgetModel { double get angleOffset => _angleOffset; - set angleOffset(value) { + set angleOffset(double value) { _angleOffset = value; refresh(); } @@ -85,9 +85,6 @@ class YAGSLSwerveDriveModel extends NTWidgetModel { @override List getEditProperties(BuildContext context) { - String angleUnit = - tryCast(ntConnection.getLastAnnouncedValue(rotationUnitTopic)) ?? - "radians"; return [ Row( children: [ @@ -117,9 +114,13 @@ class YAGSLSwerveDriveModel extends NTWidgetModel { Flexible( child: DialogTextInput( initialText: _angleOffset.toString(), - label: 'Angle Offset ($angleUnit)', + label: 'Angle Offset (degrees)', onSubmit: (String value) async { - angleOffset = double.parse(value); + double? doubleValue = double.tryParse(value); + + if (doubleValue != null) { + angleOffset = doubleValue; + } }, formatter: TextFormatterBuilder.decimalTextFormatter(), ), @@ -212,9 +213,15 @@ class YAGSLSwerveDrive extends NTWidget { .getLastAnnouncedValue(model.rotationUnitTopic)) ?? 'radians'; + double angleOffsetConverted = model.angleOffset; + + if (rotationUnit == "radians") { + angleOffsetConverted *= (pi / 180); + } + double robotAngle = tryCast(model.ntConnection .getLastAnnouncedValue(model.robotRotationTopic)) ?? - 0.0 + model.angleOffset; + 0.0 + angleOffsetConverted; if (rotationUnit == 'degrees') { robotAngle = radians(robotAngle); From 7c534337bbda5bb9cf90ff369aefe82edf02d58d Mon Sep 17 00:00:00 2001 From: Gold87 <91761103+Gold872@users.noreply.github.com> Date: Sat, 24 Aug 2024 18:21:57 -0400 Subject: [PATCH 7/9] Apply angle offset after unit conversion --- .../nt_widgets/multi-topic/yagsl_swerve_drive.dart | 10 +++------- 1 file changed, 3 insertions(+), 7 deletions(-) diff --git a/lib/widgets/nt_widgets/multi-topic/yagsl_swerve_drive.dart b/lib/widgets/nt_widgets/multi-topic/yagsl_swerve_drive.dart index ed8f7fe4..fbd5d5d6 100644 --- a/lib/widgets/nt_widgets/multi-topic/yagsl_swerve_drive.dart +++ b/lib/widgets/nt_widgets/multi-topic/yagsl_swerve_drive.dart @@ -213,15 +213,9 @@ class YAGSLSwerveDrive extends NTWidget { .getLastAnnouncedValue(model.rotationUnitTopic)) ?? 'radians'; - double angleOffsetConverted = model.angleOffset; - - if (rotationUnit == "radians") { - angleOffsetConverted *= (pi / 180); - } - double robotAngle = tryCast(model.ntConnection .getLastAnnouncedValue(model.robotRotationTopic)) ?? - 0.0 + angleOffsetConverted; + 0.0; if (rotationUnit == 'degrees') { robotAngle = radians(robotAngle); @@ -229,6 +223,8 @@ class YAGSLSwerveDrive extends NTWidget { robotAngle *= 2 * pi; } + robotAngle += radians(model.angleOffset); + double maxSpeed = tryCast(model.ntConnection .getLastAnnouncedValue(model.maxSpeedTopic)) ?? 4.5; From 767a3ea963fc1443e8a62c10d199c70ce906f184 Mon Sep 17 00:00:00 2001 From: Gold87 <91761103+Gold872@users.noreply.github.com> Date: Sat, 24 Aug 2024 18:31:37 -0400 Subject: [PATCH 8/9] Minor formatting fixes --- .../multi-topic/yagsl_swerve_drive.dart | 7 ++++--- .../multi-topic/yagsl_swerve_drive_test.dart | 15 ++++++++------- 2 files changed, 12 insertions(+), 10 deletions(-) diff --git a/lib/widgets/nt_widgets/multi-topic/yagsl_swerve_drive.dart b/lib/widgets/nt_widgets/multi-topic/yagsl_swerve_drive.dart index fbd5d5d6..feebc57b 100644 --- a/lib/widgets/nt_widgets/multi-topic/yagsl_swerve_drive.dart +++ b/lib/widgets/nt_widgets/multi-topic/yagsl_swerve_drive.dart @@ -113,16 +113,17 @@ class YAGSLSwerveDriveModel extends NTWidgetModel { children: [ Flexible( child: DialogTextInput( - initialText: _angleOffset.toString(), + initialText: angleOffset.toString(), label: 'Angle Offset (degrees)', - onSubmit: (String value) async { + onSubmit: (String value) { double? doubleValue = double.tryParse(value); if (doubleValue != null) { angleOffset = doubleValue; } }, - formatter: TextFormatterBuilder.decimalTextFormatter(), + formatter: TextFormatterBuilder.decimalTextFormatter( + allowNegative: true), ), ), ], diff --git a/test/widgets/nt_widgets/multi-topic/yagsl_swerve_drive_test.dart b/test/widgets/nt_widgets/multi-topic/yagsl_swerve_drive_test.dart index ac23b06c..cc3abd01 100644 --- a/test/widgets/nt_widgets/multi-topic/yagsl_swerve_drive_test.dart +++ b/test/widgets/nt_widgets/multi-topic/yagsl_swerve_drive_test.dart @@ -53,13 +53,14 @@ void main() { test('YAGSL swerve drive to json', () { YAGSLSwerveDriveModel yagslSwerveModel = YAGSLSwerveDriveModel( - ntConnection: ntConnection, - preferences: preferences, - topic: 'Test/YAGSL Swerve Drive', - period: 0.100, - showRobotRotation: true, - showDesiredStates: true, - angleOffset: 90.0); + ntConnection: ntConnection, + preferences: preferences, + topic: 'Test/YAGSL Swerve Drive', + period: 0.100, + showRobotRotation: true, + showDesiredStates: true, + angleOffset: 90.0, + ); expect(yagslSwerveModel.toJson(), yagslSwerveJson); }); From 2720d3d61e82ba0645fa08eb4e5630aed91873e2 Mon Sep 17 00:00:00 2001 From: DanPeled <98838880+DanPeled@users.noreply.github.com> Date: Mon, 26 Aug 2024 21:46:23 +0300 Subject: [PATCH 9/9] negated offset --- lib/widgets/nt_widgets/multi-topic/yagsl_swerve_drive.dart | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/lib/widgets/nt_widgets/multi-topic/yagsl_swerve_drive.dart b/lib/widgets/nt_widgets/multi-topic/yagsl_swerve_drive.dart index feebc57b..d42c355d 100644 --- a/lib/widgets/nt_widgets/multi-topic/yagsl_swerve_drive.dart +++ b/lib/widgets/nt_widgets/multi-topic/yagsl_swerve_drive.dart @@ -224,7 +224,7 @@ class YAGSLSwerveDrive extends NTWidget { robotAngle *= 2 * pi; } - robotAngle += radians(model.angleOffset); + robotAngle -= radians(model.angleOffset); double maxSpeed = tryCast(model.ntConnection .getLastAnnouncedValue(model.maxSpeedTopic)) ??