mirror of
https://github.com/wpilibsuite/allwpilib
synced 2026-06-19 00:41:43 +00:00
Storage is now nested. Separate "roots" can be configured which save to separate files. In particular, this is used to save wpigui and ImGui window position to a -window.json file. ImGui's ini (for window position) is mapped to JSON. You can optionally specify a directory to load from on the command line. If one isn't provided, it uses the global system directory. Any changes made are automatically saved here. Workspace | Open: select directory, the current layout is replaced with that workspace, and future auto-saves also switch to that location. The main window size/location is not changed, only the contents. Workspace | Save As: select directory, the current layout is saved there, and future auto-saves also switch to that location. Workspace | Reset: window locations are preserved, but all other settings are reset to default (including e.g. removing plot windows). This will also end up clearing the current save file. as with load, the main window size/location is not changed. Workspace | Save As Global: "save as" to the global system location Notably, the main window size/location is only loaded at startup, but is auto-saved as part of the current workspace.
160 lines
4.6 KiB
C++
160 lines
4.6 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.
|
|
|
|
#include "glass/hardware/PCM.h"
|
|
|
|
#include <cstdio>
|
|
#include <cstring>
|
|
|
|
#include <imgui.h>
|
|
#include <wpi/SmallVector.h>
|
|
|
|
#include "glass/Context.h"
|
|
#include "glass/DataSource.h"
|
|
#include "glass/Storage.h"
|
|
#include "glass/other/DeviceTree.h"
|
|
#include "glass/support/ExtraGuiWidgets.h"
|
|
#include "glass/support/NameSetting.h"
|
|
|
|
using namespace glass;
|
|
|
|
bool glass::DisplayPCMSolenoids(PCMModel* model, int index,
|
|
bool outputsEnabled) {
|
|
wpi::SmallVector<int, 16> channels;
|
|
model->ForEachSolenoid([&](SolenoidModel& solenoid, int j) {
|
|
if (auto data = solenoid.GetOutputData()) {
|
|
if (j >= static_cast<int>(channels.size())) {
|
|
channels.resize(j + 1);
|
|
}
|
|
channels[j] = (outputsEnabled && data->GetValue()) ? 1 : -1;
|
|
}
|
|
});
|
|
|
|
if (channels.empty()) {
|
|
return false;
|
|
}
|
|
|
|
// show nonexistent channels as empty
|
|
for (auto&& ch : channels) {
|
|
if (ch == 0) {
|
|
ch = -2;
|
|
}
|
|
}
|
|
|
|
// build header label
|
|
std::string& name = GetStorage().GetString("name");
|
|
char label[128];
|
|
if (!name.empty()) {
|
|
std::snprintf(label, sizeof(label), "%s [%d]###name", name.c_str(), index);
|
|
} else {
|
|
std::snprintf(label, sizeof(label), "PCM[%d]###name", index);
|
|
}
|
|
|
|
// header
|
|
bool open = CollapsingHeader(label);
|
|
|
|
PopupEditName("name", &name);
|
|
|
|
ImGui::SetItemAllowOverlap();
|
|
ImGui::SameLine();
|
|
|
|
// show channels as LED indicators
|
|
static const ImU32 colors[] = {IM_COL32(255, 255, 102, 255),
|
|
IM_COL32(128, 128, 128, 255)};
|
|
DrawLEDs(channels.data(), channels.size(), channels.size(), colors);
|
|
|
|
if (open) {
|
|
ImGui::PushItemWidth(ImGui::GetFontSize() * 4);
|
|
model->ForEachSolenoid([&](SolenoidModel& solenoid, int j) {
|
|
if (auto data = solenoid.GetOutputData()) {
|
|
PushID(j);
|
|
char label[64];
|
|
NameSetting name{data->GetName()};
|
|
name.GetLabel(label, sizeof(label), "Solenoid", j);
|
|
data->LabelText(label, "%s", channels[j] == 1 ? "On" : "Off");
|
|
name.PopupEditName(j);
|
|
PopID();
|
|
}
|
|
});
|
|
ImGui::PopItemWidth();
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
void glass::DisplayPCMsSolenoids(PCMsModel* model, bool outputsEnabled,
|
|
std::string_view noneMsg) {
|
|
bool hasAny = false;
|
|
model->ForEachPCM([&](PCMModel& pcm, int i) {
|
|
PushID(i);
|
|
if (DisplayPCMSolenoids(&pcm, i, outputsEnabled)) {
|
|
hasAny = true;
|
|
}
|
|
PopID();
|
|
});
|
|
if (!hasAny && !noneMsg.empty()) {
|
|
ImGui::TextUnformatted(noneMsg.data(), noneMsg.data() + noneMsg.size());
|
|
}
|
|
}
|
|
|
|
void glass::DisplayCompressorDevice(PCMModel* model, int index,
|
|
bool outputsEnabled) {
|
|
auto compressor = model->GetCompressor();
|
|
if (!compressor || !compressor->Exists()) {
|
|
return;
|
|
}
|
|
DisplayCompressorDevice(compressor, index, outputsEnabled);
|
|
}
|
|
|
|
void glass::DisplayCompressorDevice(CompressorModel* model, int index,
|
|
bool outputsEnabled) {
|
|
char name[32];
|
|
std::snprintf(name, sizeof(name), "Compressor[%d]", index);
|
|
if (BeginDevice(name)) {
|
|
// output enabled
|
|
if (auto runningData = model->GetRunningData()) {
|
|
bool value = outputsEnabled && runningData->GetValue();
|
|
if (DeviceBoolean("Running", false, &value, runningData)) {
|
|
model->SetRunning(value);
|
|
}
|
|
}
|
|
|
|
// closed loop enabled
|
|
if (auto enabledData = model->GetEnabledData()) {
|
|
int value = enabledData->GetValue() ? 1 : 0;
|
|
static const char* enabledOptions[] = {"disabled", "enabled"};
|
|
if (DeviceEnum("Closed Loop", true, &value, enabledOptions, 2,
|
|
enabledData)) {
|
|
model->SetEnabled(value != 0);
|
|
}
|
|
}
|
|
|
|
// pressure switch
|
|
if (auto pressureSwitchData = model->GetPressureSwitchData()) {
|
|
int value = pressureSwitchData->GetValue() ? 1 : 0;
|
|
static const char* switchOptions[] = {"full", "low"};
|
|
if (DeviceEnum("Pressure", false, &value, switchOptions, 2,
|
|
pressureSwitchData)) {
|
|
model->SetPressureSwitch(value != 0);
|
|
}
|
|
}
|
|
|
|
// compressor current
|
|
if (auto currentData = model->GetCurrentData()) {
|
|
double value = currentData->GetValue();
|
|
if (DeviceDouble("Current (A)", false, &value, currentData)) {
|
|
model->SetCurrent(value);
|
|
}
|
|
}
|
|
|
|
EndDevice();
|
|
}
|
|
}
|
|
|
|
void glass::DisplayCompressorsDevice(PCMsModel* model, bool outputsEnabled) {
|
|
model->ForEachPCM([&](PCMModel& pcm, int i) {
|
|
DisplayCompressorDevice(&pcm, i, outputsEnabled);
|
|
});
|
|
}
|