15using namespace std::literals;
20static constexpr std::array<uint16_t, 16> fixedPalette = {
21 0x000, 0x000, 0x611, 0x733,
22 0x117, 0x327, 0x151, 0x627,
23 0x171, 0x373, 0x661, 0x664,
24 0x411, 0x265, 0x555, 0x777,
30 customPalette = fixedPalette;
37 buf.append(
"customPalette=");
38 for (
auto i :
xrange(16)) {
39 if (i) buf.append(
",");
40 buf.appendf(
"%d", customPalette[i]);
49 }
else if (name ==
"customPalette") {
52 if (
auto v = StringOp::stringTo<uint16_t>(s)) {
53 customPalette[i] = *v;
60static std::array<int, 3> extractRGB(uint16_t grb)
62 auto r = (grb >> 4) & 7;
63 auto g = (grb >> 8) & 7;
64 auto b = (grb >> 0) & 7;
68static void insertRGB(std::span<uint16_t, 16> msxPalette,
unsigned index, std::array<int, 3> rgb)
71 msxPalette[index] = narrow<int16_t>(((rgb[1] & 7) << 8) | ((rgb[0] & 7) << 4) | ((rgb[2] & 7) << 0));
74[[nodiscard]]
static uint32_t toRGBA(uint16_t msxColor)
76 auto [r,
g, b] = extractRGB(msxColor);
77 static constexpr float scale = 1.0f / 7.0f;
78 return ImColor(
float(r) * scale,
float(
g) * scale,
float(b) * scale);
81static bool coloredButton(
const char*
id, ImU32 color, ImVec2 size)
84 auto col = ImGui::ColorConvertU32ToFloat4(color);
86 ImGuiCol_ButtonHovered, col,
87 ImGuiCol_ButtonActive, col, [&]{
88 result = ImGui::Button(
id, size);
95 std::array<uint32_t, 16> result;
98 for (
auto i :
xrange(16)) {
99 auto [r,
g, b] = rgb[i];
100 result[i] = ImColor(r,
g, b);
103 auto rgb = [&]() -> std::span<const uint16_t, 16> {
105 return customPalette;
112 for (
auto i :
xrange(16)) {
113 result[i] = toRGBA(rgb[i]);
123 VDP* vdp = motherBoard ?
dynamic_cast<VDP*
>(motherBoard->
findDevice(
"VDP")) :
nullptr;
125 ImGui::RadioButton(
"VDP palette", &
whichPalette, PALETTE_VDP);
127 ImGui::RadioButton(
"Custom palette", &
whichPalette, PALETTE_CUSTOM);
129 ImGui::RadioButton(
"Fixed palette", &
whichPalette, PALETTE_FIXED);
131 std::array<uint16_t, 16> paletteCopy;
132 std::span<uint16_t, 16> palette = customPalette;
136 palette = std::span<uint16_t, 16>{
const_cast<uint16_t*
>(fixedPalette.data()), 16};
140 palette = paletteCopy;
144 ImGui::TableSetupColumn(
"left", ImGuiTableColumnFlags_WidthFixed, 200.0f);
145 ImGui::TableSetupColumn(
"right", ImGuiTableColumnFlags_WidthFixed, 150.0f);
147 if (ImGui::TableNextColumn()) {
148 std::array<uint32_t, 16> paletteRGB =
getPalette(vdp);
151 if (ImGui::TableNextColumn()) {
152 if (i == selectedColor) {
153 ImGui::TableSetBgColor(ImGuiTableBgTarget_CellBg, ImGui::GetColorU32(ImGuiCol_HeaderActive));
155 if (coloredButton(
"", paletteRGB[i], {44.0f, 30.0f})) {
162 if (ImGui::TableNextColumn()) {
166 coloredButton(
"##color", toRGBA(palette[selectedColor]), {150.0f, 30.0f});
171 static constexpr std::array names = {
"R"sv,
"G"sv,
"B"sv};
172 auto rgb = extractRGB(palette[selectedColor]);
174 ImGui::AlignTextToFramePadding();
178 ImGuiCol_FrameBg,
static_cast<ImVec4
>(ImColor::HSV(
float(i) * (1.0f / 3.0f), 0.5f, 0.5f)),
179 ImGuiCol_FrameBgHovered,
static_cast<ImVec4
>(ImColor::HSV(
float(i) * (1.0f / 3.0f), 0.6f, 0.5f)),
180 ImGuiCol_FrameBgActive,
static_cast<ImVec4
>(ImColor::HSV(
float(i) * (1.0f / 3.0f), 0.7f, 0.5f)),
181 ImGuiCol_SliderGrab,
static_cast<ImVec4
>(ImColor::HSV(
float(i) * (1.0f / 3.0f), 0.9f, 0.9f)), [&]{
182 ImGui::SetNextItemWidth(-FLT_MIN);
183 if (ImGui::SliderInt(
"##v", &rgb[i], 0, 7,
"%d", ImGuiSliderFlags_AlwaysClamp)) {
185 insertRGB(palette, selectedColor, rgb);
188 vdp->
setPalette(selectedColor, palette[selectedColor], time);
void save(ImGuiTextBuffer &buf) override
std::array< uint32_t, 16 > getPalette(const VDP *vdp) const
void loadLine(std::string_view name, zstring_view value) override
void paint(MSXMotherBoard *motherBoard) override
ImGuiPalette(ImGuiManager &manager_)
EmuTime::param getCurrentTime() const
Convenience method: This is the same as getScheduler().getCurrentTime().
MSXDevice * findDevice(std::string_view name)
Find a MSXDevice by name.
Unified implementation of MSX Video Display Processors (VDPs).
std::array< std::array< uint8_t, 3 >, 16 > getMSX1Palette() const
Get the (fixed) palette for this MSX1 VDP.
void setPalette(unsigned index, word grb, EmuTime::param time)
Sets a palette entry.
uint16_t getPalette(unsigned index) const
Gets a palette entry.
bool isMSX1VDP() const
Is this an MSX1 VDP?
Like std::string_view, but with the extra guarantee that it refers to a zero-terminated string.
void TextUnformatted(const std::string &str)
auto split_view(std::string_view str, Separators separators)
constexpr mat4 scale(const vec3 &xyz)
void Table(const char *str_id, int column, ImGuiTableFlags flags, const ImVec2 &outer_size, float inner_width, std::invocable<> auto next)
void Window(const char *name, bool *p_open, ImGuiWindowFlags flags, std::invocable<> auto next)
void StyleColor(bool active, Args &&...args)
void Disabled(bool b, std::invocable<> auto next)
void ID_for_range(std::integral auto count, std::invocable< int > auto next)
This file implemented 3 utility functions:
bool loadOnePersistent(std::string_view name, zstring_view value, C &c, const std::tuple< Elements... > &tup)
void savePersistent(ImGuiTextBuffer &buf, C &c, const std::tuple< Elements... > &tup)
constexpr auto copy(InputRange &&range, OutputIter out)
constexpr auto xrange(T e)