mirror of
https://github.com/wpilibsuite/allwpilib
synced 2026-06-22 01:11:42 +00:00
[wpimath] Add fmtlib formatter overloads for Eigen::Matrix and units (#3409)
This allows using Eigen matrices or units natively with fmt::format() or fmt::print().
This commit is contained in:
44
wpimath/src/main/native/include/frc/fmt/Eigen.h
Normal file
44
wpimath/src/main/native/include/frc/fmt/Eigen.h
Normal file
@@ -0,0 +1,44 @@
|
||||
// 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 <fmt/format.h>
|
||||
|
||||
#include "Eigen/Core"
|
||||
|
||||
template <int Rows, int Cols, int... Args>
|
||||
struct fmt::formatter<Eigen::Matrix<double, Rows, Cols, Args...>> {
|
||||
char presentation = 'f';
|
||||
|
||||
constexpr auto parse(fmt::format_parse_context& ctx) {
|
||||
auto it = ctx.begin(), end = ctx.end();
|
||||
if (it != end && (*it == 'f' || *it == 'e')) {
|
||||
presentation = *it++;
|
||||
}
|
||||
|
||||
if (it != end && *it != '}') {
|
||||
throw fmt::format_error("invalid format");
|
||||
}
|
||||
|
||||
return it;
|
||||
}
|
||||
|
||||
template <typename FormatContext>
|
||||
auto format(const Eigen::Matrix<double, Rows, Cols, Args...>& mat,
|
||||
FormatContext& ctx) {
|
||||
auto out = ctx.out();
|
||||
for (int i = 0; i < Rows; ++i) {
|
||||
for (int j = 0; j < Cols; ++j) {
|
||||
out = fmt::format_to(out, " {:f}", mat(i, j));
|
||||
}
|
||||
|
||||
if (i < Rows - 1) {
|
||||
out = fmt::format_to(out, "\n");
|
||||
}
|
||||
}
|
||||
|
||||
return out;
|
||||
}
|
||||
};
|
||||
212
wpimath/src/main/native/include/frc/fmt/Units.h
Normal file
212
wpimath/src/main/native/include/frc/fmt/Units.h
Normal file
@@ -0,0 +1,212 @@
|
||||
// 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 <fmt/format.h>
|
||||
|
||||
#include "units/base.h"
|
||||
|
||||
template <class Units, typename T, template <typename> class NonLinearScale>
|
||||
struct fmt::formatter<units::unit_t<Units, T, NonLinearScale>> {
|
||||
char presentation = 'f';
|
||||
|
||||
constexpr auto parse(fmt::format_parse_context& ctx) {
|
||||
auto it = ctx.begin(), end = ctx.end();
|
||||
if (it != end && (*it == 'f' || *it == 'e')) {
|
||||
presentation = *it++;
|
||||
}
|
||||
|
||||
if (it != end && *it != '}') {
|
||||
throw fmt::format_error("invalid format");
|
||||
}
|
||||
|
||||
return it;
|
||||
}
|
||||
|
||||
template <typename FormatContext>
|
||||
auto format(const units::unit_t<Units, T, NonLinearScale>& obj,
|
||||
FormatContext& ctx) {
|
||||
using BaseUnits =
|
||||
units::unit<std::ratio<1>,
|
||||
typename units::traits::unit_traits<Units>::base_unit_type>;
|
||||
|
||||
auto out = ctx.out();
|
||||
|
||||
out = fmt::format_to(out, "{}", units::convert<Units, BaseUnits>(obj()));
|
||||
|
||||
if constexpr (units::traits::unit_traits<
|
||||
Units>::base_unit_type::meter_ratio::num != 0) {
|
||||
out = fmt::format_to(out, " m");
|
||||
}
|
||||
if constexpr (units::traits::unit_traits<
|
||||
Units>::base_unit_type::meter_ratio::num != 0 &&
|
||||
units::traits::unit_traits<
|
||||
Units>::base_unit_type::meter_ratio::num != 1) {
|
||||
out = fmt::format_to(
|
||||
out, "^{}",
|
||||
units::traits::unit_traits<Units>::base_unit_type::meter_ratio::num);
|
||||
}
|
||||
if constexpr (units::traits::unit_traits<
|
||||
Units>::base_unit_type::meter_ratio::den != 1) {
|
||||
out = fmt::format_to(
|
||||
out, "/{}",
|
||||
units::traits::unit_traits<Units>::base_unit_type::meter_ratio::den);
|
||||
}
|
||||
|
||||
if constexpr (units::traits::unit_traits<
|
||||
Units>::base_unit_type::kilogram_ratio::num != 0) {
|
||||
out = fmt::format_to(out, " kg");
|
||||
}
|
||||
if constexpr (units::traits::unit_traits<
|
||||
Units>::base_unit_type::kilogram_ratio::num != 0 &&
|
||||
units::traits::unit_traits<
|
||||
Units>::base_unit_type::kilogram_ratio::num != 1) {
|
||||
out = fmt::format_to(out, "^{}",
|
||||
units::traits::unit_traits<
|
||||
Units>::base_unit_type::kilogram_ratio::num);
|
||||
}
|
||||
if constexpr (units::traits::unit_traits<
|
||||
Units>::base_unit_type::kilogram_ratio::den != 1) {
|
||||
out = fmt::format_to(out, "/{}",
|
||||
units::traits::unit_traits<
|
||||
Units>::base_unit_type::kilogram_ratio::den);
|
||||
}
|
||||
|
||||
if constexpr (units::traits::unit_traits<
|
||||
Units>::base_unit_type::second_ratio::num != 0) {
|
||||
out = fmt::format_to(out, " s");
|
||||
}
|
||||
if constexpr (units::traits::unit_traits<
|
||||
Units>::base_unit_type::second_ratio::num != 0 &&
|
||||
units::traits::unit_traits<
|
||||
Units>::base_unit_type::second_ratio::num != 1) {
|
||||
out = fmt::format_to(
|
||||
out, "^{}",
|
||||
units::traits::unit_traits<Units>::base_unit_type::second_ratio::num);
|
||||
}
|
||||
if constexpr (units::traits::unit_traits<
|
||||
Units>::base_unit_type::second_ratio::den != 1) {
|
||||
out = fmt::format_to(
|
||||
out, "/{}",
|
||||
units::traits::unit_traits<Units>::base_unit_type::second_ratio::den);
|
||||
}
|
||||
|
||||
if constexpr (units::traits::unit_traits<
|
||||
Units>::base_unit_type::ampere_ratio::num != 0) {
|
||||
out = fmt::format_to(out, " A");
|
||||
}
|
||||
if constexpr (units::traits::unit_traits<
|
||||
Units>::base_unit_type::ampere_ratio::num != 0 &&
|
||||
units::traits::unit_traits<
|
||||
Units>::base_unit_type::ampere_ratio::num != 1) {
|
||||
out = fmt::format_to(
|
||||
out, "^{}",
|
||||
units::traits::unit_traits<Units>::base_unit_type::ampere_ratio::num);
|
||||
}
|
||||
if constexpr (units::traits::unit_traits<
|
||||
Units>::base_unit_type::ampere_ratio::den != 1) {
|
||||
out = fmt::format_to(
|
||||
out, "/{}",
|
||||
units::traits::unit_traits<Units>::base_unit_type::ampere_ratio::den);
|
||||
}
|
||||
|
||||
if constexpr (units::traits::unit_traits<
|
||||
Units>::base_unit_type::kelvin_ratio::num != 0) {
|
||||
out = fmt::format_to(out, " K");
|
||||
}
|
||||
if constexpr (units::traits::unit_traits<
|
||||
Units>::base_unit_type::kelvin_ratio::num != 0 &&
|
||||
units::traits::unit_traits<
|
||||
Units>::base_unit_type::kelvin_ratio::num != 1) {
|
||||
out = fmt::format_to(
|
||||
out, "^{}",
|
||||
units::traits::unit_traits<Units>::base_unit_type::kelvin_ratio::num);
|
||||
}
|
||||
if constexpr (units::traits::unit_traits<
|
||||
Units>::base_unit_type::kelvin_ratio::den != 1) {
|
||||
out = fmt::format_to(
|
||||
out, "/{}",
|
||||
units::traits::unit_traits<Units>::base_unit_type::kelvin_ratio::den);
|
||||
}
|
||||
|
||||
if constexpr (units::traits::unit_traits<
|
||||
Units>::base_unit_type::mole_ratio::num != 0) {
|
||||
out = fmt::format_to(out, " mol");
|
||||
}
|
||||
if constexpr (units::traits::unit_traits<
|
||||
Units>::base_unit_type::mole_ratio::num != 0 &&
|
||||
units::traits::unit_traits<
|
||||
Units>::base_unit_type::mole_ratio::num != 1) {
|
||||
out = fmt::format_to(
|
||||
out, "^{}",
|
||||
units::traits::unit_traits<Units>::base_unit_type::mole_ratio::num);
|
||||
}
|
||||
if constexpr (units::traits::unit_traits<
|
||||
Units>::base_unit_type::mole_ratio::den != 1) {
|
||||
out = fmt::format_to(
|
||||
out, "/{}",
|
||||
units::traits::unit_traits<Units>::base_unit_type::mole_ratio::den);
|
||||
}
|
||||
|
||||
if constexpr (units::traits::unit_traits<
|
||||
Units>::base_unit_type::candela_ratio::num != 0) {
|
||||
out = fmt::format_to(out, " cd");
|
||||
}
|
||||
if constexpr (units::traits::unit_traits<
|
||||
Units>::base_unit_type::candela_ratio::num != 0 &&
|
||||
units::traits::unit_traits<
|
||||
Units>::base_unit_type::candela_ratio::num != 1) {
|
||||
out = fmt::format_to(out, "^{}",
|
||||
units::traits::unit_traits<
|
||||
Units>::base_unit_type::candela_ratio::num);
|
||||
}
|
||||
if constexpr (units::traits::unit_traits<
|
||||
Units>::base_unit_type::candela_ratio::den != 1) {
|
||||
out = fmt::format_to(out, "/{}",
|
||||
units::traits::unit_traits<
|
||||
Units>::base_unit_type::candela_ratio::den);
|
||||
}
|
||||
|
||||
if constexpr (units::traits::unit_traits<
|
||||
Units>::base_unit_type::radian_ratio::num != 0) {
|
||||
out = fmt::format_to(out, " rad");
|
||||
}
|
||||
if constexpr (units::traits::unit_traits<
|
||||
Units>::base_unit_type::radian_ratio::num != 0 &&
|
||||
units::traits::unit_traits<
|
||||
Units>::base_unit_type::radian_ratio::num != 1) {
|
||||
out = fmt::format_to(
|
||||
out, "^{}",
|
||||
units::traits::unit_traits<Units>::base_unit_type::radian_ratio::num);
|
||||
}
|
||||
if constexpr (units::traits::unit_traits<
|
||||
Units>::base_unit_type::radian_ratio::den != 1) {
|
||||
out = fmt::format_to(
|
||||
out, "/{}",
|
||||
units::traits::unit_traits<Units>::base_unit_type::radian_ratio::den);
|
||||
}
|
||||
|
||||
if constexpr (units::traits::unit_traits<
|
||||
Units>::base_unit_type::byte_ratio::num != 0) {
|
||||
out = fmt::format_to(out, " b");
|
||||
}
|
||||
if constexpr (units::traits::unit_traits<
|
||||
Units>::base_unit_type::byte_ratio::num != 0 &&
|
||||
units::traits::unit_traits<
|
||||
Units>::base_unit_type::byte_ratio::num != 1) {
|
||||
out = fmt::format_to(
|
||||
out, "^{}",
|
||||
units::traits::unit_traits<Units>::base_unit_type::byte_ratio::num);
|
||||
}
|
||||
if constexpr (units::traits::unit_traits<
|
||||
Units>::base_unit_type::byte_ratio::den != 1) {
|
||||
out = fmt::format_to(
|
||||
out, "/{}",
|
||||
units::traits::unit_traits<Units>::base_unit_type::byte_ratio::den);
|
||||
}
|
||||
|
||||
return out;
|
||||
}
|
||||
};
|
||||
Reference in New Issue
Block a user