mirror of
https://github.com/wpilibsuite/allwpilib
synced 2026-06-29 02:21:44 +00:00
This refactors Alert in both c++ and java to fix the issues with the current c++ implementation and improve performance. Currently, constructing an Alert adds it to a list of Alerts with the same group and type. Activating an alert sets a flag on the alert. When the SendableAlerts is polled (GetStrings), the entire list is iterated over, filtered, and the filtered list is sorted by timestamp. This leads to a worst case O(m + nlog(n)) time complexity for GetStrings, where m and n are the count of total constructed alerts active alerts respectively. It also allocates intermediate data structures to hold the active alert strings for sorting. This changes the implementation to improve the performance of GetStrings, by shifting the sorting overhead to Alert.Set Constructing the Alert only initializes the alert's initial state, and initializes the SendableAlerts for the group if it is not already initialized. Activating or deactivating an alert sets an internal flag for state tracking, and inserts or removes a structure containing the timestamp and text into a self-sorting data structure (std::set, TreeSet) containing other active alerts with the same group and type. (worst case O(log(n)) Now, SendableAlerts.GetStrings only has to iterate over the structure and copy the strings to the returned array. (amortized O(n)) This also fixes the c++ implementation by removing the need for SendableAlerts to directly access the Alert. This also adds a helper method to SendableRegistry to force initialization of the instance to prevent static initialization ordering issues.
142 lines
3.9 KiB
C++
142 lines
3.9 KiB
C++
// Copyright (c) FIRST and other WPILib contributors.
|
|
// Open Source Software; you can modify and/or share it under the terms of
|
|
// the WPILib BSD license file in the root directory of this project.
|
|
|
|
#pragma once
|
|
|
|
#include <stdint.h>
|
|
|
|
#include <set>
|
|
#include <string>
|
|
|
|
namespace frc {
|
|
|
|
/**
|
|
* Persistent alert to be sent via NetworkTables. Alerts are tagged with a type
|
|
* of kError, kWarning, or kInfo to denote urgency. See Alert::AlertType for
|
|
* suggested usage of each type. Alerts can be displayed on supported
|
|
* dashboards, and are shown in a priority order based on type and recency of
|
|
* activation, with newly activated alerts first.
|
|
*
|
|
* Alerts should be created once and stored persistently, then updated to
|
|
* "active" or "inactive" as necessary. Set(bool) can be safely called
|
|
* periodically.
|
|
*
|
|
* This API is new for 2025, but is likely to change in future seasons to
|
|
* facilitate deeper integration with the robot control system.
|
|
*
|
|
* <pre>
|
|
* class Robot {
|
|
* frc::Alert alert{"Something went wrong", frc::Alert::AlertType::kWarning};
|
|
* }
|
|
*
|
|
* Robot::periodic() {
|
|
* alert.Set(...);
|
|
* }
|
|
* </pre>
|
|
*/
|
|
class Alert {
|
|
public:
|
|
/**
|
|
* Represents an alert's level of urgency.
|
|
*/
|
|
enum class AlertType {
|
|
/**
|
|
* High priority alert - displayed first on the dashboard with a red "X"
|
|
* symbol. Use this type for problems which will seriously affect the
|
|
* robot's functionality and thus require immediate attention.
|
|
*/
|
|
kError,
|
|
|
|
/**
|
|
* Medium priority alert - displayed second on the dashboard with a yellow
|
|
* "!" symbol. Use this type for problems which could affect the robot's
|
|
* functionality but do not necessarily require immediate attention.
|
|
*/
|
|
kWarning,
|
|
|
|
/**
|
|
* Low priority alert - displayed last on the dashboard with a green "i"
|
|
* symbol. Use this type for problems which are unlikely to affect the
|
|
* robot's functionality, or any other alerts which do not fall under the
|
|
* other categories.
|
|
*/
|
|
kInfo
|
|
};
|
|
|
|
/**
|
|
* Creates a new alert in the default group - "Alerts". If this is the first
|
|
* to be instantiated, the appropriate entries will be added to NetworkTables.
|
|
*
|
|
* @param text Text to be displayed when the alert is active.
|
|
* @param type Alert urgency level.
|
|
*/
|
|
Alert(std::string_view text, AlertType type);
|
|
|
|
/**
|
|
* Creates a new alert. If this is the first to be instantiated in its group,
|
|
* the appropriate entries will be added to NetworkTables.
|
|
*
|
|
* @param group Group identifier, used as the entry name in NetworkTables.
|
|
* @param text Text to be displayed when the alert is active.
|
|
* @param type Alert urgency level.
|
|
*/
|
|
Alert(std::string_view group, std::string_view text, AlertType type);
|
|
|
|
Alert(Alert&&);
|
|
Alert& operator=(Alert&&);
|
|
|
|
Alert(const Alert&) = default;
|
|
Alert& operator=(const Alert&) = default;
|
|
|
|
~Alert();
|
|
|
|
/**
|
|
* Sets whether the alert should currently be displayed. This method can be
|
|
* safely called periodically.
|
|
*
|
|
* @param active Whether to display the alert.
|
|
*/
|
|
void Set(bool active);
|
|
|
|
/**
|
|
* Gets whether the alert is active.
|
|
* @return whether the alert is active.
|
|
*/
|
|
bool Get() const { return m_active; }
|
|
|
|
/**
|
|
* Updates current alert text. Use this method to dynamically change the
|
|
* displayed alert, such as including more details about the detected problem.
|
|
*
|
|
* @param text Text to be displayed when the alert is active.
|
|
*/
|
|
void SetText(std::string_view text);
|
|
|
|
/**
|
|
* Gets the current alert text.
|
|
* @return the current text.
|
|
*/
|
|
std::string GetText() const { return m_text; }
|
|
|
|
/**
|
|
* Get the type of this alert.
|
|
* @return the type
|
|
*/
|
|
AlertType GetType() const { return m_type; }
|
|
|
|
private:
|
|
class PublishedAlert;
|
|
class SendableAlerts;
|
|
|
|
AlertType m_type;
|
|
std::string m_text;
|
|
std::set<PublishedAlert>* m_activeAlerts;
|
|
bool m_active = false;
|
|
uint64_t m_activeStartTime;
|
|
};
|
|
|
|
std::string format_as(Alert::AlertType type);
|
|
|
|
} // namespace frc
|