vst3 ui resize on linux seems to work now

This commit is contained in:
Stefano D'Angelo 2024-05-10 17:54:13 +02:00
parent e2d244f4e7
commit bb463ccc7b
5 changed files with 63 additions and 13 deletions

View File

@ -65,6 +65,14 @@ LDFLAGS_ALL := ${LDFLAGS_ALL} -arch arm64 -arch x86_64
CXXFLAGS_ALL := ${CXXFLAGS_ALL} -arch arm64 -arch x86_64 CXXFLAGS_ALL := ${CXXFLAGS_ALL} -arch arm64 -arch x86_64
endif endif
ifeq ($(UNAME_S), Linux)
ifeq (${HAS_UI}, yes)
CFLAGS_ALL := ${CFLAGS_ALL} $(shell pkg-config --cflags x11)
LDFLAGS_ALL := ${LDFLAGS_ALL} $(shell pkg-config --libs x11)
CXXFLAGS_ALL := ${CXXFLAGS_ALL} $(shell pkg-config --cflags x11)
endif
endif
BUNDLE_DIR = ${BUNDLE_NAME}.vst3 BUNDLE_DIR = ${BUNDLE_NAME}.vst3
DLL_DIR = Contents/${PLATFORM} DLL_DIR = Contents/${PLATFORM}

View File

@ -30,3 +30,5 @@ CXX_SRCS_EXTRA := {{=it.make && it.make.cxxSrcs ? it.make.cxxSrcs : ""}} {{=it.v
COMMON_DIR := {{=it.vst3_make && it.vst3_make.commonDir ? it.vst3_make.commonDir : (it.make && it.make.commonDir ? it.make.commonDir : "")}} COMMON_DIR := {{=it.vst3_make && it.vst3_make.commonDir ? it.vst3_make.commonDir : (it.make && it.make.commonDir ? it.make.commonDir : "")}}
DATA_DIR := {{=it.vst3_make && it.vst3_make.dataDir ? it.vst3_make.dataDir : (it.make && it.make.dataDir ? it.make.dataDir : "")}} DATA_DIR := {{=it.vst3_make && it.vst3_make.dataDir ? it.vst3_make.dataDir : (it.make && it.make.dataDir ? it.make.dataDir : "")}}
PLUGIN_DIR := {{=it.vst3_make && it.vst3_make.pluginDir ? it.vst3_make.pluginDir : (it.make && it.make.pluginDir ? it.make.pluginDir : "")}} PLUGIN_DIR := {{=it.vst3_make && it.vst3_make.pluginDir ? it.vst3_make.pluginDir : (it.make && it.make.pluginDir ? it.make.pluginDir : "")}}
HAS_UI := {{=it.product.ui ? "yes" : "no"}}

View File

@ -218,3 +218,7 @@ static struct {
{{?it.product.parameters.find(x => x.isLatency)}} {{?it.product.parameters.find(x => x.isLatency)}}
#define DATA_PARAM_LATENCY_INDEX {{=it.product.parameters.find(x => x.isLatency).paramIndex}} #define DATA_PARAM_LATENCY_INDEX {{=it.product.parameters.find(x => x.isLatency).paramIndex}}
{{?}} {{?}}
{{?it.product.ui}}
#define DATA_UI_USER_RESIZABLE {{=it.product.ui.userResizable ? 1 : 0}}
{{?}}

View File

@ -42,7 +42,7 @@
#ifdef TIBIA_TRACE #ifdef TIBIA_TRACE
# include <stdio.h> # include <stdio.h>
# define TRACE(...) printf(__VA_ARGS__) # define TRACE(...) printf(__VA_ARGS__); fflush(stdout);
#else #else
# define TRACE(...) /* do nothing */ # define TRACE(...) /* do nothing */
#endif #endif
@ -839,6 +839,8 @@ static Steinberg_Vst_IProcessContextRequirementsVtbl pluginVtblIProcessContextRe
#ifdef PLUGIN_UI #ifdef PLUGIN_UI
# ifdef __linux__ # ifdef __linux__
# include <X11/Xlib.h>
typedef struct { typedef struct {
Steinberg_ITimerHandlerVtbl * vtblITimerHandler; Steinberg_ITimerHandlerVtbl * vtblITimerHandler;
Steinberg_uint32 refs; Steinberg_uint32 refs;
@ -907,15 +909,16 @@ typedef struct plugView {
Steinberg_IPlugViewVtbl * vtblIPlugView; Steinberg_IPlugViewVtbl * vtblIPlugView;
Steinberg_uint32 refs; Steinberg_uint32 refs;
Steinberg_IPlugFrame * frame; Steinberg_IPlugFrame * frame;
plugin_ui * ui;
# ifdef __linux__ # ifdef __linux__
Steinberg_IRunLoop * runLoop; Steinberg_IRunLoop * runLoop;
timerHandler timer; timerHandler timer;
Display * display;
# endif # endif
plugin_ui * ui;
} plugView; } plugView;
static Steinberg_tresult plugViewQueryInterface(void *thisInterface, const Steinberg_TUID iid, void ** obj) { static Steinberg_tresult plugViewQueryInterface(void *thisInterface, const Steinberg_TUID iid, void ** obj) {
TRACE("plugView IEditController queryInterface %p\n", thisInterface); TRACE("plugView queryInterface %p\n", thisInterface);
// Same as above (pluginQueryInterface) // Same as above (pluginQueryInterface)
size_t offset; size_t offset;
if (memcmp(iid, Steinberg_FUnknown_iid, sizeof(Steinberg_TUID)) == 0 if (memcmp(iid, Steinberg_FUnknown_iid, sizeof(Steinberg_TUID)) == 0
@ -936,7 +939,7 @@ static Steinberg_tresult plugViewQueryInterface(void *thisInterface, const Stein
} }
static Steinberg_uint32 plugViewAddRef(void *thisInterface) { static Steinberg_uint32 plugViewAddRef(void *thisInterface) {
TRACE("plugView IEditController addRef %p\n", thisInterface); TRACE("plugView addRef %p\n", thisInterface);
plugView *v = (plugView *)((char *)thisInterface - offsetof(plugView, vtblIPlugView)); plugView *v = (plugView *)((char *)thisInterface - offsetof(plugView, vtblIPlugView));
v->refs++; v->refs++;
return v->refs; return v->refs;
@ -948,6 +951,9 @@ static Steinberg_uint32 plugViewRelease(void *thisInterface) {
v->refs--; v->refs--;
if (v->refs == 0) { if (v->refs == 0) {
TRACE(" free %p\n", (void *)v); TRACE(" free %p\n", (void *)v);
# ifdef __linux__
XCloseDisplay(v->display);
# endif
free(v); free(v);
return 0; return 0;
} }
@ -978,7 +984,7 @@ static Steinberg_tresult plugViewAttached(void* thisInterface, void* parent, Ste
if (!v->ui) if (!v->ui)
return Steinberg_kResultFalse; return Steinberg_kResultFalse;
# ifdef __linux # ifdef __linux
if (v->runLoop->lpVtbl->registerTimer(v->runLoop, &v->timer, 20) != Steinberg_kResultOk) { if (v->runLoop->lpVtbl->registerTimer(v->runLoop, (struct Steinberg_ITimerHandler *)&v->timer, 20) != Steinberg_kResultOk) {
plugin_ui_free(v->ui); plugin_ui_free(v->ui);
return Steinberg_kResultFalse; return Steinberg_kResultFalse;
} }
@ -989,8 +995,9 @@ static Steinberg_tresult plugViewAttached(void* thisInterface, void* parent, Ste
static Steinberg_tresult plugViewRemoved(void* thisInterface) { static Steinberg_tresult plugViewRemoved(void* thisInterface) {
TRACE("plugView removed %p\n", thisInterface); TRACE("plugView removed %p\n", thisInterface);
plugView *v = (plugView *)((char *)thisInterface - offsetof(plugView, vtblIPlugView)); plugView *v = (plugView *)((char *)thisInterface - offsetof(plugView, vtblIPlugView));
v->runLoop->lpVtbl->unregisterTimer(v->runLoop, &v->timer); v->runLoop->lpVtbl->unregisterTimer(v->runLoop, (struct Steinberg_ITimerHandler *)&v->timer);
plugin_ui_free(v->ui); plugin_ui_free(v->ui);
v->ui = NULL;
return Steinberg_kResultTrue; return Steinberg_kResultTrue;
} }
@ -1016,11 +1023,24 @@ static Steinberg_tresult plugViewGetSize(void* thisInterface, struct Steinberg_V
TRACE("plugView getSize %p %p\n", thisInterface, size); TRACE("plugView getSize %p %p\n", thisInterface, size);
if (!size) if (!size)
return Steinberg_kInvalidArgument; return Steinberg_kInvalidArgument;
//TODO plugView *v = (plugView *)((char *)thisInterface - offsetof(plugView, vtblIPlugView));
size->left = 0; size->left = 0;
size->top = 0; size->top = 0;
size->right = 600; if (!v->ui) {
size->bottom = 400; uint32_t width, height;
plugin_ui_get_default_size(&width, &height);
size->right = width;
size->bottom = height;
} else {
# ifdef __linux__
XWindowAttributes attr;
TRACE(" window %u\n", (Window)(*((char **)v->ui)));
XGetWindowAttributes(v->display, (Window)(*((char **)v->ui)), &attr);
size->right = attr.width;
size->bottom = attr.height;
# endif
}
TRACE(" %u x %u\n", size->right, size->bottom);
return Steinberg_kResultTrue; return Steinberg_kResultTrue;
} }
@ -1041,17 +1061,22 @@ static Steinberg_tresult plugViewSetFrame(void* thisInterface, struct Steinberg_
plugView *v = (plugView *)((char *)thisInterface - offsetof(plugView, vtblIPlugView)); plugView *v = (plugView *)((char *)thisInterface - offsetof(plugView, vtblIPlugView));
v->frame = frame; v->frame = frame;
# ifdef __linux__ # ifdef __linux__
if (v->frame->lpVtbl->queryInterface(v->frame, Steinberg_IRunLoop_iid, (void **)&v->runLoop) != Steinberg_kResultOk) if (v->frame) {
return Steinberg_kResultFalse; if (v->frame->lpVtbl->queryInterface(v->frame, Steinberg_IRunLoop_iid, (void **)&v->runLoop) != Steinberg_kResultOk)
v->frame->lpVtbl->release(v->frame); return Steinberg_kResultFalse;
v->frame->lpVtbl->release(v->frame);
}
# endif # endif
return Steinberg_kResultTrue; return Steinberg_kResultTrue;
} }
static Steinberg_tresult plugViewCanResize(void* thisInterface) { static Steinberg_tresult plugViewCanResize(void* thisInterface) {
TRACE("plugView canResize %p\n", thisInterface); TRACE("plugView canResize %p\n", thisInterface);
//TODO # if DATA_UI_USER_RESIZABLE
return Steinberg_kResultTrue;
# else
return Steinberg_kResultFalse; return Steinberg_kResultFalse;
# endif
} }
static Steinberg_tresult plugViewCheckSizeConstraint(void* thisInterface, struct Steinberg_ViewRect* rect) { static Steinberg_tresult plugViewCheckSizeConstraint(void* thisInterface, struct Steinberg_ViewRect* rect) {
@ -1422,11 +1447,17 @@ static struct Steinberg_IPlugView* controllerCreateView(void* thisInterface, Ste
view->vtblIPlugView = &plugViewVtblIPlugView; view->vtblIPlugView = &plugViewVtblIPlugView;
view->refs = 1; view->refs = 1;
view->frame = NULL; view->frame = NULL;
view->ui = NULL;
# ifdef __linux__ # ifdef __linux__
view->timer.vtblITimerHandler = &timerHandlerVtblITimerHandler; view->timer.vtblITimerHandler = &timerHandlerVtblITimerHandler;
view->timer.refs = 1; view->timer.refs = 1;
view->timer.cb = plugViewOnTimer; view->timer.cb = plugViewOnTimer;
view->timer.data = view; view->timer.data = view;
view->display = XOpenDisplay(NULL);
if (view->display == NULL) {
free(view);
return NULL;
}
# endif # endif
return (struct Steinberg_IPlugView *)view; return (struct Steinberg_IPlugView *)view;

View File

@ -129,6 +129,11 @@ typedef struct {
PuglView * view; PuglView * view;
} plugin_ui; } plugin_ui;
static void plugin_ui_get_default_size(uint32_t *width, uint32_t *height) {
*width = 600;
*height = 400;
}
static PuglStatus plugin_ui_on_event(PuglView *view, const PuglEvent *event) { static PuglStatus plugin_ui_on_event(PuglView *view, const PuglEvent *event) {
switch (event->type) { switch (event->type) {
case PUGL_EXPOSE: case PUGL_EXPOSE: