Build release version, separate UI into header file

This commit is contained in:
yaw-man 2022-08-18 12:42:48 -03:00
parent d4644df706
commit 4c8cd87421
2 changed files with 265 additions and 226 deletions

View File

@ -1,5 +1,4 @@
#include "DistrhoUI.hpp" #include "ui.h"
#include "tablet.h"
#ifdef DEBUG #ifdef DEBUG
#include <format> #include <format>
#include <optional> #include <optional>
@ -7,259 +6,231 @@
START_NAMESPACE_DISTRHO START_NAMESPACE_DISTRHO
//UI element for mapping the currently pressed pen button combination to a parameter. ButtonMappingWidget::ButtonMappingWidget(
class ButtonMappingWidget : public NanoSubWidget, Widget *parent,
public ButtonEventHandler float initialSize,
float initialX,
float initialY,
Parameters associatedParameter,
ButtonEventHandler::Callback *const callback) : x(initialX),
y(initialY),
size(initialSize),
param(associatedParameter),
isClicked(false),
isPenPressed(false),
mask(0),
NanoSubWidget(parent),
ButtonEventHandler(this)
{ {
public: setSize(Size<uint>(static_cast<uint>(size), static_cast<uint>(size)));
ButtonMappingWidget( setAbsolutePos((int)x, (int)y);
Widget* parent, ButtonEventHandler::setCallback(callback);
float initialSize, }
float initialX,
float initialY,
Parameters associatedParameter,
ButtonEventHandler::Callback* const callback) :
x(initialX),
y(initialY),
size(initialSize),
param(associatedParameter),
isClicked(false),
isPenPressed(false),
mask(0),
NanoSubWidget(parent),
ButtonEventHandler(this)
{
setSize(Size<uint>(static_cast<uint>(size), static_cast<uint>(size)));
setAbsolutePos((int)x, (int)y);
ButtonEventHandler::setCallback(callback);
}
void onNanoDisplay() override void ButtonMappingWidget::onNanoDisplay()
{
beginPath();
strokeColor(200, 200, 200);
fillColor(0.5f, 0.5f, 0.5f, 0.5f * (isClicked + isPenPressed));
roundedRect(0.f, 0.f, size, size, 0.5f * size);
stroke();
fill();
closePath();
}
bool onMouse(const MouseEvent& ev) override
{
isClicked = ev.press;
return ButtonEventHandler::mouseEvent(ev);
}
void setMask(const ButtonMask m) { mask = m; }
bool matchesMask(const ButtonMask m) { return isPenPressed = (m && mask == m); }
bool isClicked;
bool isPenPressed;
private:
float x = 0;
float y = 0;
float size = 0;
const Parameters param;
ButtonMask mask;
};
class TabUI : public UI,
public ButtonEventHandler::Callback
{ {
static const uint kInitialWidth = 800; beginPath();
static const uint kInitialHeight = 600; strokeColor(200, 200, 200);
fillColor(0.5f, 0.5f, 0.5f, 0.5f * (isClicked + isPenPressed));
roundedRect(0.f, 0.f, size, size, 0.5f * size);
stroke();
fill();
closePath();
}
public: bool ButtonMappingWidget::onMouse(const MouseEvent &ev)
TabUI() {
: UI(kInitialWidth, kInitialHeight), isClicked = ev.press;
tab(getWindow().getNativeWindowHandle()), return ButtonEventHandler::mouseEvent(ev);
AButtonWidget(this, 75.f, 700.f, 500.f, kParameterButtonA, this), }
BButtonWidget(this, 75.f, 700.f, 400.f, kParameterButtonB, this)
{ static constexpr uint kInitialWidth = 800;
static constexpr uint kInitialHeight = 600;
TabUI::TabUI()
: UI(kInitialWidth, kInitialHeight),
tab(getWindow().getNativeWindowHandle()),
AButtonWidget(this, 75.f, 700.f, 500.f, kParameterButtonA, this),
BButtonWidget(this, 75.f, 700.f, 400.f, kParameterButtonB, this)
{
#ifdef DGL_NO_SHARED_RESOURCES #ifdef DGL_NO_SHARED_RESOURCES
createFontFromFile("sans", "/usr/share/fonts/truetype/ttf-dejavu/DejaVuSans.ttf"); createFontFromFile("sans", "/usr/share/fonts/truetype/ttf-dejavu/DejaVuSans.ttf");
#else #else
loadSharedResources(); loadSharedResources();
#endif #endif
if (!tab.initialized) return; if (!tab.initialized)
//setGeometryConstraints(400, static_cast<uint>(300 * tabletAspectRatio), true, false); return;
} // setGeometryConstraints(400, static_cast<uint>(300 * tabletAspectRatio), true, false);
}
protected: void TabUI::getTabletData()
{
if (!tab.initialized || !tab.GetPacket(pkt))
return;
if (pkt == lastPkt)
return;
if (pkt.x != lastPkt.x)
setParameterValue(ktpax, pkt.x);
if (pkt.y != lastPkt.y)
setParameterValue(ktpay, pkt.y);
if (pkt.z != lastPkt.z)
setParameterValue(ktpaz, pkt.z);
if (pkt.p != lastPkt.p)
setParameterValue(ktpap, pkt.p);
if (pkt.buttons != lastPkt.buttons)
setButtonsValue(pkt.buttons);
lastPkt = pkt;
}
void getTabletData() void TabUI::setButtonsValue(unsigned long buttonMask)
{
if (AButtonWidget.matchesMask(buttonMask))
setParameterValue(kParameterButtonA, 1.f);
else
setParameterValue(kParameterButtonA, 0.f);
if (BButtonWidget.matchesMask(buttonMask))
setParameterValue(kParameterButtonB, 1.f);
else
setParameterValue(kParameterButtonB, 0.f);
}
void TabUI::buttonClicked(SubWidget *const widget, int)
{
if (widget == &AButtonWidget)
AButtonWidget.setMask(pkt.buttons);
if (widget == &BButtonWidget)
BButtonWidget.setMask(pkt.buttons);
}
void TabUI::parameterChanged(uint32_t index, float value)
{
if (index != kParameterTime && index < kParameterCount)
{ {
if (!tab.initialized || !tab.GetPacket(pkt)) return; switch (index)
if (pkt == lastPkt) return; {
if (pkt.x != lastPkt.x) setParameterValue(ktpax, pkt.x); case (ktpax):
if (pkt.y != lastPkt.y) setParameterValue(ktpay, pkt.y); pkt.x = value;
if (pkt.z != lastPkt.z) setParameterValue(ktpaz, pkt.z); case (ktpay):
if (pkt.p != lastPkt.p) setParameterValue(ktpap, pkt.p); pkt.y = value;
if (pkt.buttons != lastPkt.buttons) setButtonsValue(pkt.buttons); case (ktpaz):
lastPkt = pkt; pkt.z = value;
} case (ktpap):
pkt.p = value;
void setButtonsValue(unsigned long buttonMask)
{
if (AButtonWidget.matchesMask(buttonMask))
setParameterValue(kParameterButtonA, 1.f);
else setParameterValue(kParameterButtonA, 0.f);
if (BButtonWidget.matchesMask(buttonMask))
setParameterValue(kParameterButtonB, 1.f);
else setParameterValue(kParameterButtonB, 0.f);
}
void buttonClicked(SubWidget* const widget, int ) override
{
if (widget == &AButtonWidget) AButtonWidget.setMask(pkt.buttons);
if (widget == &BButtonWidget) BButtonWidget.setMask(pkt.buttons);
}
void parameterChanged(uint32_t index, float value) override
{
if (index != kParameterTime && index < kParameterCount) {
switch (index)
{
case(ktpax): pkt.x = value;
case(ktpay): pkt.y = value;
case(ktpaz): pkt.z = value;
case(ktpap): pkt.p = value;
}
return;
} }
return;
getTabletData();
repaint();
} }
void uiIdle() override getTabletData();
repaint();
}
void TabUI::uiIdle()
{
getTabletData();
repaint();
}
bool TabUI::onMouse(const MouseEvent &ev)
{
getTabletData();
repaint();
return false; // Allow event to propagate.
}
bool TabUI::onScroll(const ScrollEvent &ev)
{
double add;
const uint x = getWidth();
const uint y = getHeight();
add = (ev.delta.getY() > 0) ? 1 : -1;
float tabletAspectRatio;
if (tab.initialized)
tabletAspectRatio = tab.ext.y / tab.ext.x;
else
tabletAspectRatio = 1.f;
setSize(static_cast<uint>(x + add + 0.5), static_cast<uint>(tabletAspectRatio * (x + add + 0.5)));
return true;
}
void TabUI::onNanoDisplay()
{
fontSize(15.0f);
textLineHeight(1.f);
// Report tablet errors.
#ifdef DEBUG
if (!tab.initialized)
{ {
getTabletData(); const std::string err = std::vformat("Tablet not supported:\n{}",
repaint(); std::make_format_args(tab.errormsg));
}
bool onMouse(const MouseEvent& ev) override
{
getTabletData();
repaint();
return false; //Allow event to propagate.
}
bool onScroll(const ScrollEvent& ev) override
{
double add;
const uint x = getWidth();
const uint y = getHeight();
add = (ev.delta.getY() > 0) ? 1 : -1;
float tabletAspectRatio;
if (tab.initialized) tabletAspectRatio = tab.ext.y / tab.ext.x; else tabletAspectRatio = 1.f;
setSize(static_cast<uint>(x + add + 0.5), static_cast<uint>(tabletAspectRatio * (x + add + 0.5)));
return true;
}
void onNanoDisplay() override
{
fontSize(15.0f);
textLineHeight(1.f);
//Report tablet errors.
if (!tab.initialized) {
const std::string err = std::vformat("Tablet not supported:\n{}",
std::make_format_args(tab.errormsg));
beginPath();
fillColor(200, 200, 200);
textBox(0.f, 15.f, 250.f, err.c_str(), nullptr);
closePath();
return;
}
//Numerical feedback.
beginPath(); beginPath();
fillColor(200, 200, 200); fillColor(200, 200, 200);
textBox(0.f, 15.f, 250.f, textBox(0.f, 15.f, 250.f, err.c_str(), nullptr);
closePath();
return;
}
// Numerical feedback.
beginPath();
fillColor(200, 200, 200);
textBox(0.f, 15.f, 250.f,
std::format("x: {:.3f}\ny: {:.3f}\nz: {:.3f}\np: {:.3f}\nb: {}", std::format("x: {:.3f}\ny: {:.3f}\nz: {:.3f}\np: {:.3f}\nb: {}",
pkt.x, pkt.y, pkt.z, pkt.p, pkt.buttons).c_str(), nullptr); pkt.x, pkt.y, pkt.z, pkt.p, pkt.buttons)
closePath(); .c_str(),
nullptr);
closePath();
#endif
//Pen position and pressure. // Pen position and pressure.
drawCircle(pkt.x, pkt.y, pkt.z, pkt.p); drawCircle(pkt.x, pkt.y, pkt.z, pkt.p);
} }
void drawCircle(float x, float y, float z, float p) { void TabUI::drawCircle(float x, float y, float z, float p)
{
static constexpr float circleRadius = 25.f; static constexpr float circleRadius = 25.f;
x *= getWidth(); x *= getWidth();
y = (1.f - y) * getHeight(); y = (1.f - y) * getHeight();
z = 1.f - z; z = 1.f - z;
beginPath(); beginPath();
strokeColor(1.f, 1.f, 1.f, 0.5f); strokeColor(1.f, 1.f, 1.f, 0.5f);
moveTo(x - z * circleRadius, y); moveTo(x - z * circleRadius, y);
lineTo(x + z * circleRadius, y); lineTo(x + z * circleRadius, y);
stroke(); stroke();
closePath(); closePath();
beginPath(); beginPath();
strokeColor(1.f, 1.f, 1.f, 0.5f); strokeColor(1.f, 1.f, 1.f, 0.5f);
moveTo(x, y - z * circleRadius); moveTo(x, y - z * circleRadius);
lineTo(x, y + z * circleRadius); lineTo(x, y + z * circleRadius);
stroke(); stroke();
closePath(); closePath();
beginPath(); beginPath();
fillColor(1.f, 1.f, 1.f, p); fillColor(1.f, 1.f, 1.f, p);
strokeColor(255, 255, 255, 255); strokeColor(255, 255, 255, 255);
circle(x, y, circleRadius); circle(x, y, circleRadius);
fill(); fill();
stroke(); stroke();
closePath(); closePath();
beginPath(); beginPath();
strokeColor(1.f, 1.f, 1.f, z); strokeColor(1.f, 1.f, 1.f, z);
circle(x, y, z * circleRadius); circle(x, y, z * circleRadius);
stroke(); stroke();
closePath(); closePath();
} }
UI *createUI()
// -------------------------------------------------------------------------------------------------------
private:
// Tablet context handler
Tablet tab;
Packet pkt = { 0 };
Packet lastPkt = { 0 };
// Button mapping widgets
ButtonMappingWidget AButtonWidget;
ButtonMappingWidget BButtonWidget;
/**
Set our UI class as non-copyable and add a leak detector just in case.
*/
DISTRHO_DECLARE_NON_COPYABLE_WITH_LEAK_DETECTOR(TabUI)
};
UI* createUI()
{ {
return new TabUI(); return new TabUI();
} }
END_NAMESPACE_DISTRHO END_NAMESPACE_DISTRHO

68
src/yaw-tab/ui.h Normal file
View File

@ -0,0 +1,68 @@
#include "DistrhoUI.hpp"
#include "tablet.h"
#ifdef DEBUG
#include <format>
#include <optional>
#endif
START_NAMESPACE_DISTRHO
class ButtonMappingWidget : public NanoSubWidget,
public ButtonEventHandler
{
public:
explicit ButtonMappingWidget(
Widget *parent,
float initialSize,
float initialX,
float initialY,
Parameters associatedParameter,
ButtonEventHandler::Callback *const callback);
void onNanoDisplay() override;
bool onMouse(const MouseEvent &ev) override;
void setMask(const ButtonMask m) { mask = m; };
bool matchesMask(const ButtonMask m) { return isPenPressed = (m && mask == m); };
bool isClicked;
bool isPenPressed;
private:
float x = 0;
float y = 0;
float size = 0;
const Parameters param;
ButtonMask mask;
};
class TabUI : public UI,
public ButtonEventHandler::Callback
{
public:
explicit TabUI();
void buttonClicked(SubWidget *const widget, int) override;
void parameterChanged(uint32_t index, float value) override;
void uiIdle() override;
bool onMouse(const MouseEvent &ev) override;
bool onScroll(const ScrollEvent &ev) override;
void onNanoDisplay() override;
private:
void getTabletData();
void setButtonsValue(unsigned long buttonMask);
void drawCircle(float x, float y, float z, float p);
// Tablet context handler
Tablet tab;
Packet pkt = {0};
Packet lastPkt = {0};
// Button mapping widgets
ButtonMappingWidget AButtonWidget;
ButtonMappingWidget BButtonWidget;
DISTRHO_DECLARE_NON_COPYABLE_WITH_LEAK_DETECTOR(TabUI)
};
END_NAMESPACE_DISTRHO