mirror of
https://github.com/wpilibsuite/allwpilib
synced 2026-06-19 00:41:43 +00:00
The HAL will remain untouched in order to maintain C-style compatibility. A few places in wpilibc were left as C-style strings, especially if special formatting (eg, elaborate uses of snprintf or sscanf) was being used. In general, const char* was changed to std::string. character buffers used for formatting were either untouched, changed to std::stringstream, or changed to std::string, depending on what was done with the buffer. Change-Id: I5e431ddf1cc4d9a6d534e1f21b16ea23be26e7f1
195 lines
6.3 KiB
C++
195 lines
6.3 KiB
C++
#include "LiveWindow/LiveWindow.h"
|
|
#include "networktables/NetworkTable.h"
|
|
#include <algorithm>
|
|
#include <sstream>
|
|
|
|
/**
|
|
* Get an instance of the LiveWindow main class
|
|
* This is a singleton to guarantee that there is only a single instance
|
|
* regardless of
|
|
* how many times GetInstance is called.
|
|
*/
|
|
LiveWindow &LiveWindow::GetInstance() {
|
|
static LiveWindow instance;
|
|
return instance;
|
|
}
|
|
|
|
/**
|
|
* LiveWindow constructor.
|
|
* Allocate the necessary tables.
|
|
*/
|
|
LiveWindow::LiveWindow() : m_scheduler(Scheduler::GetInstance()) {
|
|
m_liveWindowTable.reset(NetworkTable::GetTable("LiveWindow"));
|
|
m_statusTable.reset(m_liveWindowTable->GetSubTable("~STATUS~"));
|
|
}
|
|
|
|
/**
|
|
* Change the enabled status of LiveWindow
|
|
* If it changes to enabled, start livewindow running otherwise stop it
|
|
*/
|
|
void LiveWindow::SetEnabled(bool enabled) {
|
|
if (m_enabled == enabled) return;
|
|
if (enabled) {
|
|
if (m_firstTime) {
|
|
InitializeLiveWindowComponents();
|
|
m_firstTime = false;
|
|
}
|
|
m_scheduler.SetEnabled(false);
|
|
m_scheduler.RemoveAll();
|
|
for (auto& elem : m_components) {
|
|
elem.first->StartLiveWindowMode();
|
|
}
|
|
} else {
|
|
for (auto& elem : m_components) {
|
|
elem.first->StopLiveWindowMode();
|
|
}
|
|
m_scheduler.SetEnabled(true);
|
|
}
|
|
m_enabled = enabled;
|
|
m_statusTable->PutBoolean("LW Enabled", m_enabled);
|
|
}
|
|
|
|
/**
|
|
* Add a Sensor associated with the subsystem and with call it by the given
|
|
* name.
|
|
* @param subsystem The subsystem this component is part of.
|
|
* @param name The name of this component.
|
|
* @param component A LiveWindowSendable component that represents a sensor.
|
|
*/
|
|
void LiveWindow::AddSensor(const std::string &subsystem, const std::string &name,
|
|
::std::shared_ptr<LiveWindowSendable> component) {
|
|
m_components[component].subsystem = subsystem;
|
|
m_components[component].name = name;
|
|
m_components[component].isSensor = true;
|
|
}
|
|
|
|
[[deprecated(
|
|
"Raw pointers are deprecated; pass the component using shared_ptr "
|
|
"instead.")]]
|
|
void LiveWindow::AddSensor(const std::string &subsystem, const std::string &name, LiveWindowSendable *component) {
|
|
AddSensor(subsystem, name, ::std::shared_ptr<LiveWindowSendable>(
|
|
component, NullDeleter<LiveWindowSendable>()));
|
|
}
|
|
|
|
/**
|
|
* Add an Actuator associated with the subsystem and with call it by the given
|
|
* name.
|
|
* @param subsystem The subsystem this component is part of.
|
|
* @param name The name of this component.
|
|
* @param component A LiveWindowSendable component that represents a actuator.
|
|
*/
|
|
void LiveWindow::AddActuator(const std::string &subsystem, const std::string &name,
|
|
::std::shared_ptr<LiveWindowSendable> component) {
|
|
m_components[component].subsystem = subsystem;
|
|
m_components[component].name = name;
|
|
m_components[component].isSensor = false;
|
|
}
|
|
|
|
[[deprecated(
|
|
"Raw pointers are deprecated; pass the component using shared_ptr "
|
|
"instead.")]]
|
|
void LiveWindow::AddActuator(const std::string &subsystem, const std::string &name,
|
|
LiveWindowSendable *component) {
|
|
AddActuator(subsystem, name,
|
|
::std::shared_ptr<LiveWindowSendable>(
|
|
component, NullDeleter<LiveWindowSendable>()));
|
|
}
|
|
|
|
/**
|
|
* INTERNAL
|
|
*/
|
|
[[deprecated]]
|
|
void LiveWindow::AddSensor(std::string type, int channel,
|
|
LiveWindowSendable *component) {
|
|
std::ostringstream oss;
|
|
oss << type << "[" << channel << "]";
|
|
std::string types(oss.str());
|
|
auto cc = new char[types.size() + 1];
|
|
types.copy(cc, types.size());
|
|
cc[types.size()] = '\0';
|
|
AddSensor("Ungrouped", cc, component);
|
|
::std::shared_ptr<LiveWindowSendable> component_stl(
|
|
component, NullDeleter<LiveWindowSendable>());
|
|
if (std::find(m_sensors.begin(), m_sensors.end(), component_stl) ==
|
|
m_sensors.end())
|
|
m_sensors.push_back(component_stl);
|
|
}
|
|
|
|
/**
|
|
* INTERNAL
|
|
*/
|
|
void LiveWindow::AddActuator(std::string type, int channel,
|
|
LiveWindowSendable *component) {
|
|
std::ostringstream oss;
|
|
oss << type << "[" << channel << "]";
|
|
std::string types(oss.str());
|
|
auto cc = new char[types.size() + 1];
|
|
types.copy(cc, types.size());
|
|
cc[types.size()] = '\0';
|
|
AddActuator("Ungrouped", cc, ::std::shared_ptr<LiveWindowSendable>(
|
|
component, [](LiveWindowSendable *) {}));
|
|
}
|
|
|
|
/**
|
|
* INTERNAL
|
|
*/
|
|
void LiveWindow::AddActuator(std::string type, int module, int channel,
|
|
LiveWindowSendable *component) {
|
|
std::ostringstream oss;
|
|
oss << type << "[" << module << "," << channel << "]";
|
|
std::string types(oss.str());
|
|
auto cc = new char[types.size() + 1];
|
|
types.copy(cc, types.size());
|
|
cc[types.size()] = '\0';
|
|
AddActuator("Ungrouped", cc, ::std::shared_ptr<LiveWindowSendable>(
|
|
component, [](LiveWindowSendable *) {}));
|
|
}
|
|
|
|
/**
|
|
* Tell all the sensors to update (send) their values
|
|
* Actuators are handled through callbacks on their value changing from the
|
|
* SmartDashboard widgets.
|
|
*/
|
|
void LiveWindow::UpdateValues() {
|
|
for (auto& elem : m_sensors) {
|
|
elem->UpdateTable();
|
|
}
|
|
}
|
|
|
|
/**
|
|
* This method is called periodically to cause the sensors to send new values
|
|
* to the SmartDashboard.
|
|
*/
|
|
void LiveWindow::Run() {
|
|
if (m_enabled) {
|
|
UpdateValues();
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Initialize all the LiveWindow elements the first time we enter LiveWindow
|
|
* mode. By holding off creating the NetworkTable entries, it allows them to be
|
|
* redefined before the first time in LiveWindow mode. This allows default
|
|
* sensor and actuator values to be created that are replaced with the custom
|
|
* names from users calling addActuator and addSensor.
|
|
*/
|
|
void LiveWindow::InitializeLiveWindowComponents() {
|
|
for (auto& elem : m_components) {
|
|
::std::shared_ptr<LiveWindowSendable> component = elem.first;
|
|
LiveWindowComponent c = elem.second;
|
|
std::string subsystem = c.subsystem;
|
|
std::string name = c.name;
|
|
m_liveWindowTable->GetSubTable(subsystem)
|
|
->PutString("~TYPE~", "LW Subsystem");
|
|
::std::shared_ptr<ITable> table(
|
|
m_liveWindowTable->GetSubTable(subsystem)->GetSubTable(name));
|
|
table->PutString("~TYPE~", component->GetSmartDashboardType());
|
|
table->PutString("Name", name);
|
|
table->PutString("Subsystem", subsystem);
|
|
component->InitTable(table);
|
|
if (c.isSensor) {
|
|
m_sensors.push_back(component);
|
|
}
|
|
}
|
|
}
|