commit f186c492a5085101c30d78d02eb86169fd4d202b
parent a4ecf324794fc55370f43570fdc5aac29d07ee6d
Author: fundamental <[email protected]>
Date: Fri, 8 Feb 2013 18:20:46 -0500
Minimal GUI librtosc integration
- Well, it sure isn't pretty, but it works
- GUI can now react to OSC events from the backend
- GUI layer abstracted away from main.cpp slightly
- VU meters are now updated ONLY via OSC messages passed safely between layers
- VU meters are no longer inside MasterUI.fl as working with fluid is perhaps
one of the most painful parts of dealing with complex FLTK UI
- This commit likely breaks the no-UI build for the time being
- May the vumutex warm up hell for its larger friend
Diffstat:
12 files changed, 555 insertions(+), 408 deletions(-)
diff --git a/src/Misc/Master.cpp b/src/Misc/Master.cpp
@@ -62,6 +62,12 @@ T lim(T min, T max, T val)
static Ports localports = {
{"echo", ":'hidden':Hidden port to echo messages", 0, [](const char *m, RtData) {
bToU->raw_write(m-1);}},
+ {"get-vu", "::", 0, [](const char *, RtData d) {
+ Master *m = (Master*)d.obj;
+ bToU->write("/vu-meter", "bb", sizeof(m->vu), &m->vu, sizeof(float)*NUM_MIDI_PARTS, m->vuoutpeakpart);}},
+ {"reset-vu", "::", 0, [](const char *, RtData d) {
+ Master *m = (Master*)d.obj;
+ m->vuresetpeaks();}},
PARAMF(Master, volume, volume, log, 0.01, 4.42, "Master Volume"),
};
@@ -79,7 +85,6 @@ Master::Master()
swaplr = 0;
pthread_mutex_init(&mutex, NULL);
- pthread_mutex_init(&vumutex, NULL);
fft = new FFTwrapper(synth->oscilsize);
shutup = 0;
@@ -486,10 +491,7 @@ void Master::AudioOut(float *outl, float *outr)
outr[i] *= volume;
}
- if(!pthread_mutex_trylock(&vumutex)) {
- vuUpdate(outl, outr);
- pthread_mutex_unlock(&vumutex);
- }
+ vuUpdate(outl, outr);
//Shutup if it is asked (with fade-out)
if(shutup) {
@@ -563,7 +565,6 @@ Master::~Master()
delete fft;
pthread_mutex_destroy(&mutex);
- pthread_mutex_destroy(&vumutex);
}
@@ -619,22 +620,11 @@ void Master::ShutUp()
*/
void Master::vuresetpeaks()
{
- pthread_mutex_lock(&vumutex);
vu.outpeakl = 1e-9;
vu.outpeakr = 1e-9;
vu.maxoutpeakl = 1e-9;
vu.maxoutpeakr = 1e-9;
vu.clipped = 0;
- pthread_mutex_unlock(&vumutex);
-}
-
-vuData Master::getVuData()
-{
- vuData tmp;
- pthread_mutex_lock(&vumutex);
- tmp = vu;
- pthread_mutex_unlock(&vumutex);
- return tmp;
}
void Master::applyparameters(bool lockmutex)
diff --git a/src/Misc/Master.h b/src/Misc/Master.h
@@ -144,11 +144,8 @@ class Master
//peaks for VU-meter
void vuresetpeaks();
- //get VU-meter data
- vuData getVuData();
//peaks for part VU-meters
- /**\todo synchronize this with a mutex*/
float vuoutpeakpart[NUM_MIDI_PARTS];
unsigned char fakepeakpart[NUM_MIDI_PARTS]; //this is used to compute the "peak" when the part is disabled
@@ -161,14 +158,14 @@ class Master
class FFTwrapper * fft;
pthread_mutex_t mutex;
- pthread_mutex_t vumutex;
-
static rtosc::Ports &ports;
float volume;
+
+ //Statistics on output levels
+ vuData vu;
private:
bool nullRun;
- vuData vu;
float sysefxvol[NUM_SYS_EFX][NUM_MIDI_PARTS];
float sysefxsend[NUM_SYS_EFX][NUM_SYS_EFX];
int keyshift;
diff --git a/src/UI/CMakeLists.txt b/src/UI/CMakeLists.txt
@@ -34,6 +34,7 @@ add_library(zynaddsubfx_gui STATIC
${zynaddsubfx_gui_FLTK_UI_SRCS}
NioUI.cpp
WidgetPDial.cpp
+ Connection.cpp
)
if(NtkGui)
diff --git a/src/UI/Connection.cpp b/src/UI/Connection.cpp
@@ -0,0 +1,140 @@
+#include "Connection.h"
+#include "Fl_Osc_Interface.h"
+#include "../globals.h"
+
+#include <rtosc/rtosc.h>
+#include <rtosc/ports.h>
+
+#include <FL/Fl.H>
+#include "common.H"
+#include "MasterUI.h"
+
+#ifdef NTK_GUI
+#include <FL/Fl_Shared_Image.H>
+#include <FL/Fl_Tiled_Image.H>
+#include <FL/Fl_Dial.H>
+#endif // NTK_GUI
+
+using namespace GUI;
+MasterUI *ui;
+
+Fl_Osc_Interface *osc;//TODO: the scope of this should be narrowed
+
+#ifdef NTK_GUI
+static Fl_Tiled_Image *module_backdrop;
+#endif
+
+void
+set_module_parameters ( Fl_Widget *o )
+{
+#ifdef NTK_GUI
+ o->box( FL_DOWN_FRAME );
+ o->align( o->align() | FL_ALIGN_IMAGE_BACKDROP );
+ o->color( FL_BLACK );
+ o->image( module_backdrop );
+ o->labeltype( FL_SHADOW_LABEL );
+#else
+ o->box( FL_PLASTIC_UP_BOX );
+ o->color( FL_CYAN );
+ o->labeltype( FL_EMBOSSED_LABEL );
+#endif
+}
+
+ui_handle_t GUI::createUi(Fl_Osc_Interface *osc, void *master, void *exit)
+{
+ ::osc = osc;
+#ifdef NTK_GUI
+ fl_register_images();
+
+ Fl_Dial::default_style(Fl_Dial::PIXMAP_DIAL);
+
+ if(Fl_Shared_Image *img = Fl_Shared_Image::get(PIXMAP_PATH "/knob.png"))
+ Fl_Dial::default_image(img);
+ else
+ Fl_Dial::default_image(Fl_Shared_Image::get(SOURCE_DIR "/../pixmaps/knob.png"));
+
+ if(Fl_Shared_Image *img = Fl_Shared_Image::get(PIXMAP_PATH "/window_backdrop.png"))
+ Fl::scheme_bg(new Fl_Tiled_Image(img));
+ else
+ Fl::scheme_bg(new Fl_Tiled_Image(Fl_Shared_Image::get(SOURCE_DIR "/../pixmaps/window_backdrop.png")));
+
+ if(Fl_Shared_Image *img = Fl_Shared_Image::get(PIXMAP_PATH "/module_backdrop.png"))
+ module_backdrop = new Fl_Tiled_Image(img);
+ else
+ module_backdrop = new Fl_Tiled_Image(Fl_Shared_Image::get(SOURCE_DIR "/../pixmaps/module_backdrop.png"));
+
+ Fl::background(50, 50, 50);
+ Fl::background2(70, 70, 70);
+ Fl::foreground(255, 255, 255);
+#endif
+
+ return (void*) (ui = new MasterUI((Master*)master, (int*)exit, osc));
+}
+void GUI::destroyUi(ui_handle_t ui)
+{
+ delete static_cast<MasterUI*>(ui);
+}
+
+#define BEGIN(x) {x,"",NULL,[](const char *m, rtosc::RtData d){ \
+ MasterUI *ui = static_cast<MasterUI*>(d.obj); \
+ rtosc_arg_t a0 = rtosc_argument(m,0); \
+ rtosc_arg_t a1 = rtosc_argument(m,1); \
+ rtosc_arg_t a2 = rtosc_argument(m,2); \
+ rtosc_arg_t a3 = rtosc_argument(m,3);
+#define END }},
+
+//DSL based ports
+static rtosc::Ports ports = {
+ BEGIN("show:T") {
+ ui->showUI();
+ } END
+ BEGIN("alert:s") {
+ fl_alert(a0.s);
+ } END
+ BEGIN("session-type:s") {
+ if(strcmp(a0.s,"LASH"))
+ return;
+ ui->sm_indicator1->value(1);
+ ui->sm_indicator2->value(1);
+ ui->sm_indicator1->tooltip("LASH");
+ ui->sm_indicator2->tooltip("LASH");
+ } END
+ BEGIN("save-master:s") {
+ ui->do_save_master(a0.s);
+ } END
+ BEGIN("load-master:s") {
+ ui->do_load_master(a0.s);
+ } END
+ BEGIN("vu-meter:bb") {
+ if(a0.b.len == sizeof(vuData) &&
+ a1.b.len == sizeof(float)*NUM_MIDI_PARTS) {
+ //Refresh the primary VU meters
+ ui->simplemastervu->update((vuData*)a0.b.data);
+ ui->mastervu->update((vuData*)a0.b.data);
+
+ float *partvu = (float*)a1.b.data;
+ for(int i=0; i<NUM_MIDI_PARTS; ++i)
+ ui->panellistitem[i]->partvu->update(partvu[i]);
+ }
+ } END
+};
+
+void GUI::raiseUi(ui_handle_t gui, const char *message)
+{
+ char buffer[1024];
+ ports.dispatch(buffer, sizeof(buffer), message+1, gui);
+}
+
+void GUI::raiseUi(ui_handle_t gui, const char *dest, const char *args, ...)
+{
+ char buffer[1024];
+ va_list va;
+ va_start(va,args);
+ if(rtosc_vmessage(buffer,1024,dest,args,va))
+ raiseUi(gui, buffer);
+}
+
+void GUI::tickUi(ui_handle_t)
+{
+ Fl::wait(0.02f);
+}
diff --git a/src/UI/Connection.h b/src/UI/Connection.h
@@ -0,0 +1,15 @@
+//Defines the methods of communication for the GUI
+//Expect this code to mutate into some sort of ugly beast that will slowly
+//remove the tendrils of the UI from the RT code
+
+class Fl_Osc_Interface;
+namespace GUI
+{
+typedef void *ui_handle_t;
+
+ui_handle_t createUi(Fl_Osc_Interface *osc, void *master, void *exit);
+void destroyUi(ui_handle_t);
+void raiseUi(ui_handle_t, const char *);
+void raiseUi(ui_handle_t, const char *, const char *, ...);
+void tickUi(ui_handle_t);
+};
diff --git a/src/UI/ConnectionDummy.cpp b/src/UI/ConnectionDummy.cpp
@@ -0,0 +1,18 @@
+
+ui_handle_t createUi(message_cb, void *Master, void *exit)
+{
+ return NULL;
+}
+void destroyUi(ui_handle_t)
+{
+}
+void raiseUi(ui_handle_t, const char *)
+{
+}
+void raiseUi(ui_handle_t, const char *, const char *, ...)
+{
+}
+void tickUi(ui_handle_t)
+{
+ usleep(100000);
+}
diff --git a/src/UI/Fl_Osc_Interface.h b/src/UI/Fl_Osc_Interface.h
@@ -0,0 +1,21 @@
+#pragma once
+#include <stdio.h>
+#include <string>
+using std::string;
+
+class Fl_Osc_Interface
+{
+ public:
+ //It is assumed that you want to have a registry for all of these
+ //elements
+ virtual void createLink(string s, class Fl_Osc_Widget*){printf("linking %s...\n", s.c_str());};
+ virtual void renameLink(string,string,class Fl_Osc_Widget*){};
+ virtual void removeLink(string,class Fl_Osc_Widget*){};
+
+ //Communication link
+ virtual void requestValue(string){};
+ virtual void writeValue(string s, float f){printf("%s -> %f\n",s.c_str(), f); };
+ virtual void writeValue(string, int){};
+ virtual void writeValue(string, bool){};
+ virtual void writeValue(string, string){};
+};
diff --git a/src/UI/MasterUI.fl b/src/UI/MasterUI.fl
@@ -1,278 +1,83 @@
# data file for the Fltk User Interface Designer (fluid)
-version 1.0110
+version 1.0300
header_name {.h}
code_name {.cc}
-decl {//Copyright (c) 2002-2009 Nasca Octavian Paul} {}
+decl {//Copyright (c) 2002-2009 Nasca Octavian Paul} {private local
+}
-decl {//License: GNU GPL version 2 or later} {}
+decl {//License: GNU GPL version 2 or later} {private local
+}
-decl {\#include <stdlib.h>} {public
+decl {\#include <stdlib.h>} {public local
}
-decl {\#include <stdio.h>} {public
+decl {\#include <stdio.h>} {public local
}
-decl {\#include <string.h>} {public
+decl {\#include <string.h>} {public local
}
-decl {\#include "WidgetPDial.h"} {public
+decl {\#include "WidgetPDial.h"} {public local
}
-decl {\#include "ADnoteUI.h"} {public
+decl {\#include "ADnoteUI.h"} {public local
}
-decl {\#include "SUBnoteUI.h"} {public
+decl {\#include "SUBnoteUI.h"} {public local
}
-decl {\#include "EffUI.h"} {public
+decl {\#include "EffUI.h"} {public local
}
-decl {\#include "VirKeyboard.h"} {public
+decl {\#include "VirKeyboard.h"} {public local
}
-decl {\#include "ConfigUI.h"} {public
+decl {\#include "ConfigUI.h"} {public local
}
-decl {\#include "BankUI.h"} {public
+decl {\#include "BankUI.h"} {public local
}
-decl {\#include "PartUI.h"} {public
+decl {\#include "PartUI.h"} {public local
}
-decl {\#include "MicrotonalUI.h"} {public
+decl {\#include "MicrotonalUI.h"} {public local
}
-decl {\#include "PresetsUI.h"} {public
+decl {\#include "PresetsUI.h"} {public local
}
decl {\#include "NioUI.h"} {public global
}
-decl {\#include "../Misc/Master.h"} {public
+decl {\#include "VuPartMeter.h"} {public local
}
-decl {\#include "../Misc/Part.h"} {public
+decl {\#include "VuMasterMeter.h"} {public local
}
-decl {\#include "../Misc/Util.h"} {public
+decl {\#include "../Misc/Master.h"} {public local
}
-decl {\#include "common.H"} {public
+decl {\#include "../Misc/Part.h"} {public local
}
-decl {\#if USE_NSM
-\#include "NSM.H"
-extern NSM_Client *nsm;
-\#endif} {public
+decl {\#include "../Misc/Util.h"} {public local
}
-decl {\#include "../globals.h"} {public
+decl {\#include "common.H"} {public local
}
-class VUMeter {: {public Fl_Box}
-} {
- Function {VUMeter(int x,int y, int w, int h, const char *label=0):Fl_Box(x,y,w,h,label)} {} {
- code {master=NULL;
-npart=-1;} {}
- }
- Function {init(Master *master_,int part_)} {} {
- code {//the "part_" parameters sets the part (if it is >=0), else it sets the master
-master=master_;
-label(NULL);
-npart=part_;
-olddbl=0.0;
-olddbr=0.0;
-oldrmsdbl=0.0;
-oldrmsdbr=0.0;} {}
- }
- Function {draw_master()} {} {
- code {\#define MIN_DB (-48)
-
-int ox=x(); int oy=y(); int lx=w(); int ly=h();
-
-vuData data = master->getVuData();
-
-//pthread_mutex_lock(&master->mutex);
-float dbl=rap2dB(data.outpeakl);
-float dbr=rap2dB(data.outpeakr);
-float rmsdbl=rap2dB(data.rmspeakl);
-float rmsdbr=rap2dB(data.rmspeakr);
-float maxdbl=rap2dB(data.maxoutpeakl);
-float maxdbr=rap2dB(data.maxoutpeakr);
-int clipped=data.clipped;
-//pthread_mutex_unlock(&master->mutex);
-
-dbl=(MIN_DB-dbl)/MIN_DB;
-if (dbl<0.0) dbl=0.0;
- else if (dbl>1.0)dbl=1.0;
-
-dbr=(MIN_DB-dbr)/MIN_DB;
-if (dbr<0.0) dbr=0.0;
- else if (dbr>1.0) dbr=1.0;
-
-dbl=dbl*0.4+olddbl*0.6;
-dbr=dbr*0.4+olddbr*0.6;
-
-if ( damage() & FL_DAMAGE_USER1 )
-{
- if ( olddbl == dbl && olddbr == dbr )
- return;
-}
-
-olddbl=dbl;
-olddbr=dbr;
-
-\#define VULENX (lx-35)
-\#define VULENY (ly/2-3)
-
-dbl*=VULENX;dbr*=VULENX;
-
-int idbl=(int) dbl;
-int idbr=(int) dbr;
-
-//compute RMS - start
-rmsdbl=(MIN_DB-rmsdbl)/MIN_DB;
-if (rmsdbl<0.0) rmsdbl=0.0;
- else if (rmsdbl>1.0) rmsdbl=1.0;
-
-rmsdbr=(MIN_DB-rmsdbr)/MIN_DB;
-if (rmsdbr<0.0) rmsdbr=0.0;
- else if (rmsdbr>1.0) rmsdbr=1.0;
-
-rmsdbl=rmsdbl*0.4+oldrmsdbl*0.6;
-rmsdbr=rmsdbr*0.4+oldrmsdbr*0.6;
-
-oldrmsdbl=rmsdbl;
-oldrmsdbr=rmsdbr;
-
-
-rmsdbl*=VULENX;rmsdbr*=VULENX;
-
-int irmsdbl=(int) rmsdbl;
-int irmsdbr=(int) rmsdbr;
-//compute RMS - end
-
-
-
-//draw the vu-meter lines
-//db
-fl_rectf(ox,oy,idbr,VULENY,0,200,255);
-fl_rectf(ox,oy+ly/2,idbl,VULENY,0,200,255);
-//black
-fl_rectf(ox+idbr,oy,VULENX-idbr,VULENY,0,0,0);
-fl_rectf(ox+idbl,oy+ly/2,VULENX-idbl,VULENY,0,0,0);
-
-//draw the scales
-float tmp=VULENX*1.0/MIN_DB;
-for (int i=1;i<1-MIN_DB;i++){
- int tx=VULENX+(int) (tmp*i);
- fl_rectf(ox+tx,oy,1,VULENY+ly/2,0,160,200);
- if (i%5==0) fl_rectf(ox+tx,oy,1,VULENY+ly/2,0,230,240);
- if (i%10==0) fl_rectf(ox+tx-1,oy,2,VULENY+ly/2,0,225,255);
-};
-
-//rms
-if (irmsdbr>2) fl_rectf(ox+irmsdbr-1,oy,3,VULENY,255,255,0);
-if (irmsdbl>2) fl_rectf(ox+irmsdbl-1,oy+ly/2,3,VULENY,255,255,0);
-
-
-//draw the red box if clipping has occured
-if (clipped==0) fl_rectf(ox+VULENX+2,oy+1,lx-VULENX-3,ly-4,0,0,10);
- else fl_rectf(ox+VULENX+2,oy+1,lx-VULENX-3,ly-4,250,10,10);
-
-//draw the maxdB
-fl_font(FL_HELVETICA|FL_BOLD,10);
-fl_color(255,255,255);
-char tmpstr[10];
-if ((maxdbl>MIN_DB-20)){
- snprintf((char *)&tmpstr,10,"%ddB",(int)maxdbr);
- fl_draw(tmpstr,ox+VULENX+1,oy+1,lx-VULENX-1,VULENY,FL_ALIGN_RIGHT,NULL,0);
-};
-if ((maxdbr>MIN_DB-20)){
- snprintf((char *)&tmpstr,10,"%ddB",(int)maxdbl);
- fl_draw(tmpstr,ox+VULENX+1,oy+ly/2+1,lx-VULENX-1,VULENY,FL_ALIGN_RIGHT,NULL,0);
-};} {}
- }
- Function {draw_part()} {} {
- code {\#define MIN_DB (-48)
-int ox=x(); int oy=y(); int lx=w(); int ly=h();
-
-if (!active_r()){
- pthread_mutex_lock(&master->vumutex);
- int fakedb=master->fakepeakpart[npart];
- pthread_mutex_unlock(&master->vumutex);
- fl_rectf(ox,oy,lx,ly,140,140,140);
- if (fakedb>0){
- fakedb=(int)(fakedb/255.0*ly)+4;
- fl_rectf(ox+2,oy+ly-fakedb,lx-4,fakedb,0,0,0);
- };
-
- return;
-};
-
-//draw the vu lines
-pthread_mutex_lock(&master->vumutex);
- float db=rap2dB(master->vuoutpeakpart[npart]);
-pthread_mutex_unlock(&master->vumutex);
-
-db=(MIN_DB-db)/MIN_DB;
-if (db<0.0) db=0.0;
- else if (db>1.0) db=1.0;
-
-db*=ly-2;
-
-int idb=(int) db;
-
-fl_rectf(ox,oy+ly-idb,lx,idb,0,200,255);
-fl_rectf(ox,oy,lx,ly-idb,0,0,0);
-
+decl {\#if USE_NSM
+\#include "NSM.H"
+extern NSM_Client *nsm;
+\#endif} {public local
+}
-//draw the scales
-float tmp=ly*1.0/MIN_DB;
- for (int i=1;i<1-MIN_DB;i++){
- int ty=ly+(int) (tmp*i);
- if (i%5==0) fl_rectf(ox,oy+ly-ty,lx,1,0,160,200);
- if (i%10==0) fl_rectf(ox,oy+ly-ty,lx,1,0,230,240);
-};} {}
- }
- Function {draw()} {} {
- code {if (npart>=0) draw_part();
- else draw_master();} {}
- }
- Function {tickdraw(VUMeter *o)} {return_type {static void}
- } {
- code {o->damage(FL_DAMAGE_USER1);} {}
- }
- Function {tick(void *v)} {return_type {static void}
- } {
- code {tickdraw((VUMeter *) v);
- Fl::repeat_timeout(1.0/18.0,tick,v);//18 fps} {}
- }
- Function {handle(int event)} {return_type int
- } {
- code {switch(event){
- case FL_SHOW:
- Fl::add_timeout(1.0/18.0,tick,this);
- break;
- case FL_HIDE:
- Fl::remove_timeout(tick,this);
- break;
- case FL_PUSH:
- if (npart>=0) break;
- pthread_mutex_lock(&master->mutex);
- master->vuresetpeaks();
- pthread_mutex_unlock(&master->mutex);
- break;
-};
-return(1);} {}
- }
- decl {Master *master;} {}
- decl {int npart;} {}
- decl {float olddbl,olddbr;} {}
- decl {float oldrmsdbl,oldrmsdbr;} {}
+decl {\#include "../globals.h"} {public local
}
-class SysEffSend {open : {public WidgetPDial}
+class SysEffSend {: {public WidgetPDial}
} {
Function {SysEffSend(int x,int y, int w, int h, const char *label=0):WidgetPDial(x,y,w,h,label)} {} {
code {master=NULL;
@@ -305,9 +110,12 @@ this->copy_label(tmp);} {}
return(WidgetPDial::handle(event));} {}
}
- decl {Master *master;} {}
- decl {int neff1;} {}
- decl {int neff2;} {}
+ decl {Master *master;} {private local
+ }
+ decl {int neff1;} {private local
+ }
+ decl {int neff2;} {private local
+ }
}
class Panellistitem {open : {public Fl_Group}
@@ -315,7 +123,7 @@ class Panellistitem {open : {public Fl_Group}
Function {make_window()} {open private
} {
Fl_Window panellistitem {open
- private xywh {608 711 100 260} type Double box NO_BOX
+ private xywh {620 721 100 260} type Double box NO_BOX
class Fl_Group visible
} {
Fl_Group panellistitemgroup {open
@@ -323,14 +131,13 @@ class Panellistitem {open : {public Fl_Group}
code0 {if (master->part[npart]->Penabled==0) o->deactivate();}
code1 {set_module_parameters( o );}
} {
- Fl_Group {} {
+ Fl_Group {} {open
xywh {45 65 15 110} box ENGRAVED_FRAME
} {
- Fl_Box {} {
+ Fl_Box partvu {
label {V U}
xywh {45 65 15 110} box FLAT_BOX color 0 selection_color 75 labelcolor 55 align 128
- code0 {o->init(master,npart);}
- class VUMeter
+ class VuPartMeter
}
}
Fl_Button partname {
@@ -426,9 +233,12 @@ panellistitemgroup->redraw();} {}
code {panellistitem->hide();
//delete(panellistitem);} {}
}
- decl {int npart;} {}
- decl {Master *master;} {}
- decl {BankUI *bankui;} {}
+ decl {int npart;} {private local
+ }
+ decl {Master *master;} {private local
+ }
+ decl {BankUI *bankui;} {private local
+ }
}
class MasterUI {open
@@ -450,8 +260,8 @@ if ((
config.save();
*exitprogram=1;
};
-\#endif} open
- xywh {80 370 390 525} type Double xclass zynaddsubfx visible
+\#endif}
+ xywh {92 456 390 525} type Double xclass zynaddsubfx visible
} {
Fl_Menu_Bar mastermenu {
xywh {-5 0 690 25}
@@ -715,21 +525,21 @@ master->shutup=1;
pthread_mutex_unlock(&master->mutex);}
xywh {280 29 105 53} color 90 labelfont 1
}
- Fl_Group partuigroup {open
+ Fl_Group partuigroup {
xywh {0 310 390 205}
} {
- Fl_Group partui {open selected
+ Fl_Group partui {open
xywh {0 310 383 175}
code0 {o->init(master->part[0],master,0,bankui);}
code1 {o->show();}
class PartUI
} {}
}
- Fl_Tabs {} {open
+ Fl_Tabs {} {
xywh {0 145 390 165} box UP_FRAME
} {
Fl_Group {} {
- label {System Effects} open
+ label {System Effects}
xywh {0 162 390 145} labelsize 15 align 9
} {
Fl_Counter syseffnocounter {
@@ -787,7 +597,7 @@ syseffectui->refresh(master->sysefx[nsyseff]);}
xywh {90 90 100 20} labelfont 1 labelsize 10
}
}
- Fl_Group syseffectuigroup {open
+ Fl_Group syseffectuigroup {
xywh {5 203 380 95} color 48
} {
Fl_Group syseffectui {
@@ -815,7 +625,7 @@ pthread_mutex_unlock(&master->mutex);}
}
}
Fl_Group {} {
- label {Insertion Effects} open
+ label {Insertion Effects}
xywh {0 165 390 145} labelsize 15 align 9 hide
} {
Fl_Counter inseffnocounter {
@@ -886,7 +696,7 @@ inseffectui->show();}
xywh {100 100 100 20} labelfont 1 labelsize 10
}
}
- Fl_Group inseffectuigroup {open
+ Fl_Group inseffectuigroup {
xywh {5 205 380 95} box FLAT_BOX color 48
} {
Fl_Group inseffectui {
@@ -974,14 +784,14 @@ mastermenu->redraw();}
xywh {192 66 30 15} labelfont 1 labelsize 10 deactivate
}
}
- Fl_Group {} {open
+ Fl_Group {} {
xywh {1 490 389 55}
} {
- Fl_Box {} {
+ Fl_Box mastervu {
label {VU-Meter}
xywh {5 490 380 30} box FLAT_BOX color 48 selection_color 75
- code0 {o->init(master,-1);}
- class VUMeter
+ code0 {o->init(osc);}
+ class VuMasterMeter
}
}
Fl_Check_Button nrpnbutton {
@@ -1013,7 +823,7 @@ simplenpartcounter->do_callback();}
callback {virkeyboard->show();}
tooltip {Virtual Keyboard} xywh {280 87 40 23} color 51 labelfont 1
}
- Fl_Group {} {open
+ Fl_Group {} {
xywh {85 32 55 110} box UP_FRAME
} {
Fl_Button {} {
@@ -1088,21 +898,21 @@ GNU General Public License for details.}
}
}
Fl_Window panelwindow {
- label {ZynAddSubFX Panel}
- xywh {89 59 630 635} type Double hide
+ label {ZynAddSubFX Panel} open
+ xywh {95 105 630 635} type Double visible
} {
- Fl_Scroll {} {
+ Fl_Scroll {} {open
xywh {0 5 570 310} type HORIZONTAL box THIN_UP_BOX
} {
- Fl_Pack {} {
+ Fl_Pack {} {open
xywh {5 10 560 285} type HORIZONTAL
code0 {for (int i=0;i<NUM_MIDI_PARTS/2;i++){panellistitem[i]=new Panellistitem(0,0,70,260,"");panellistitem[i]->init(master,i,bankui);}}
} {}
}
- Fl_Scroll {} {
+ Fl_Scroll {} {open
xywh {0 320 570 310} type HORIZONTAL box THIN_UP_BOX
} {
- Fl_Pack {} {
+ Fl_Pack {} {open
xywh {5 325 560 285} type HORIZONTAL
code0 {for (int i=NUM_MIDI_PARTS/2;i<NUM_MIDI_PARTS;i++){panellistitem[i]=new Panellistitem(0,0,70,260,"");panellistitem[i]->init(master,i,bankui);}}
} {}
@@ -1128,8 +938,8 @@ if (fl_choice("Exit and leave the unsaved data?","No","Yes",NULL)) {
config.save();
*exitprogram=1;
};
-\#endif} open
- xywh {283 262 600 335} type Double visible
+\#endif}
+ xywh {295 354 600 335} type Double visible
} {
Fl_Menu_Bar simplemastermenu {
xywh {0 0 690 25}
@@ -1229,7 +1039,7 @@ if (result==-10) fl_alert("Error: Could not load the file\\nbecause it is not an
}
}
}
- Fl_Group simplelistitemgroup {open
+ Fl_Group simplelistitemgroup {
private xywh {125 65 215 145} box UP_FRAME
code0 {if (master->part[npart]->Penabled==0) o->deactivate();}
} {
@@ -1354,14 +1164,14 @@ o->redraw();}
code0 {o->init(master);}
class VirKeys
}
- Fl_Group {} {open
+ Fl_Group {} {
xywh {340 30 255 185}
} {
- Fl_Tabs {} {open
+ Fl_Tabs {} {
xywh {345 35 245 175} box UP_FRAME align 18
} {
Fl_Group {} {
- label {System Effects} open
+ label {System Effects}
xywh {345 55 245 155} box UP_FRAME labelfont 1 labelsize 12 align 18
} {
Fl_Counter simplesyseffnocounter {
@@ -1420,7 +1230,7 @@ simplesyseffectui->refresh(master->sysefx[nsyseff]);}
xywh {100 100 100 20} labelfont 1 labelsize 10
}
}
- Fl_Group simplesyseffectuigroup {open
+ Fl_Group simplesyseffectuigroup {
xywh {350 95 235 95} color 48
} {
Fl_Group simplesyseffectui {
@@ -1552,14 +1362,14 @@ pthread_mutex_unlock(&master->mutex);}
}
}
}
- Fl_Group {} {open
+ Fl_Group {} {
xywh {5 300 590 30} box ENGRAVED_FRAME
} {
- Fl_Box {} {
+ Fl_Box simplemastervu {
label {VU-Meter}
xywh {5 300 590 30} box FLAT_BOX color 41 selection_color 75
- code0 {o->init(master,-1);}
- class VUMeter
+ code0 {o->init(osc);}
+ class VuMasterMeter
}
}
Fl_Dial simplemastervolumedial {
@@ -1679,9 +1489,10 @@ masterwindowlabel[99]='\\0';
masterwindow->label(&masterwindowlabel[0]);
simplemasterwindow->label(&masterwindowlabel[0]);} {}
}
- Function {MasterUI(Master *master_,int *exitprogram_)} {} {
+ Function {MasterUI(Master *master_,int *exitprogram_, class Fl_Osc_Interface *osc_)} {} {
code {master=master_;
exitprogram=exitprogram_;
+osc=osc_;
ninseff=0;
nsyseff=0;
npart=0;
@@ -1870,17 +1681,32 @@ virkeys->midich=master->part[npart]->Prcvchn;
simplerefresh();
bankui->hide();} {}
}
- decl {Master *master;} {}
- decl {MicrotonalUI *microtonalui;} {}
- decl {BankUI *bankui;} {}
- decl {int ninseff,npart;} {}
- decl {int nsyseff;} {}
- decl {int *exitprogram;} {}
- decl {SysEffSend *syseffsend[NUM_SYS_EFX][NUM_SYS_EFX];} {}
- decl {VirKeyboard *virkeyboard;} {}
- decl {ConfigUI *configui;} {}
- decl {int swapefftype;} {}
- decl {char masterwindowlabel[100];} {}
- decl {Panellistitem *panellistitem[NUM_MIDI_PARTS];} {}
- decl {NioUI nioui;} {}
+ decl {Master *master;} {private local
+ }
+ decl {MicrotonalUI *microtonalui;} {private local
+ }
+ decl {BankUI *bankui;} {private local
+ }
+ decl {int ninseff,npart;} {private local
+ }
+ decl {int nsyseff;} {private local
+ }
+ decl {int *exitprogram;} {private local
+ }
+ decl {SysEffSend *syseffsend[NUM_SYS_EFX][NUM_SYS_EFX];} {private local
+ }
+ decl {VirKeyboard *virkeyboard;} {private local
+ }
+ decl {ConfigUI *configui;} {private local
+ }
+ decl {int swapefftype;} {private local
+ }
+ decl {char masterwindowlabel[100];} {private local
+ }
+ decl {Panellistitem *panellistitem[NUM_MIDI_PARTS];} {selected public local
+ }
+ decl {NioUI nioui;} {private local
+ }
+ decl {class Fl_Osc_Interface *osc;} {private local
+ }
}
diff --git a/src/UI/VuMasterMeter.h b/src/UI/VuMasterMeter.h
@@ -0,0 +1,130 @@
+#include "VuMeter.h"
+#include "Fl_Osc_Interface.h"
+#define MIN_DB (-48)
+
+class VuMasterMeter: public VuMeter
+{
+ public:
+ VuMasterMeter(int x,int y, int w, int h, const char *label=0)
+ :VuMeter(x,y,w,h,label),osc(NULL)
+ {}
+
+ void init(Fl_Osc_Interface *_osc)
+ {
+ osc = _osc;
+ }
+
+ int handle(int event)
+ {
+ switch(event){
+ case FL_SHOW:
+ Fl::add_timeout(1.0/18.0,tick,osc);
+ break;
+ case FL_HIDE:
+ Fl::remove_timeout(tick,osc);
+ break;
+ case FL_PUSH:
+ osc->requestValue("/reset-vu");
+ break;
+ }
+
+ return 1;
+ }
+
+ static void tick(void *v)
+ {
+ Fl::repeat_timeout(1.0/18.0,tick,v);//18 fps
+ Fl_Osc_Interface *osc = (Fl_Osc_Interface*)v;
+ osc->requestValue("/get-vu");
+ }
+
+ void draw(void)
+ {
+ const int X = x(), Y = y(), W = w(), H = h();
+
+#define VULENX (W-35)
+#define VULENY (H/2-3)
+
+ const int idbl = dbl*VULENX;
+ const int idbr = dbr*VULENX;
+ const int irmsdbl = rmsdbl*VULENX;
+ const int irmsdbr = rmsdbr*VULENX;
+
+ //draw the vu-meter lines
+ //dB
+ fl_rectf(X,Y,idbr,VULENY,0,200,255);
+ fl_rectf(X,Y+H/2,idbl,VULENY,0,200,255);
+ //black
+ fl_rectf(X+idbr,Y,VULENX-idbr,VULENY,0,0,0);
+ fl_rectf(X+idbl,Y+H/2,VULENX-idbl,VULENY,0,0,0);
+
+ //draw the scales
+ const float tmp=VULENX*1.0/MIN_DB;
+ for (int i=1;i<1-MIN_DB;i++){
+ const int tx=VULENX+(int) (tmp*i);
+ fl_rectf(X+tx,Y,1,VULENY+H/2,0,160,200);
+ if (i%5==0) fl_rectf(X+tx,Y,1,VULENY+H/2,0,230,240);
+ if (i%10==0) fl_rectf(X+tx-1,Y,2,VULENY+H/2,0,225,255);
+ }
+
+ //rms
+ if (irmsdbr>2) fl_rectf(X+irmsdbr-1,Y,3,VULENY,255,255,0);
+ if (irmsdbl>2) fl_rectf(X+irmsdbl-1,Y+H/2,3,VULENY,255,255,0);
+
+
+ //draw the red box if clipping has occured
+ if (clipped==0) fl_rectf(X+VULENX+2,Y+1,W-VULENX-3,H-4,0,0,10);
+ else fl_rectf(X+VULENX+2,Y+1,W-VULENX-3,H-4,250,10,10);
+
+ //draw the maxdB
+ fl_font(FL_HELVETICA|FL_BOLD,10);
+ fl_color(255,255,255);
+
+ char tmpstr[10];
+ if(maxdbl>MIN_DB-20){
+ snprintf((char *)&tmpstr,10,"%ddB",(int)maxdbr);
+ fl_draw(tmpstr,X+VULENX+1,Y+1,W-VULENX-1,VULENY,FL_ALIGN_RIGHT,NULL,0);
+ }
+ if(maxdbr>MIN_DB-20){
+ snprintf((char *)&tmpstr,10,"%ddB",(int)maxdbl);
+ fl_draw(tmpstr,X+VULENX+1,Y+H/2+1,W-VULENX-1,VULENY,FL_ALIGN_RIGHT,NULL,0);
+ }
+ }
+
+ void update(vuData *d)
+ {
+ //Update properties
+ dbl = limit((MIN_DB-rap2dB(d->outpeakl))/MIN_DB);
+ dbr = limit((MIN_DB-rap2dB(d->outpeakr))/MIN_DB);
+ rmsdbl = limit((MIN_DB-rap2dB(d->rmspeakl))/MIN_DB);
+ rmsdbr = limit((MIN_DB-rap2dB(d->rmspeakr))/MIN_DB);
+ maxdbl = rap2dB(d->maxoutpeakl);
+ maxdbr = rap2dB(d->maxoutpeakr);
+ clipped = d->clipped;
+
+ //interpolate
+ dbl = 0.4 * dbl + 0.6 * olddbl;
+ dbr = 0.4 * dbr + 0.6 * olddbr;
+
+ rmsdbl = 0.4 * rmsdbl + 0.6 * oldrmsdbl;
+ rmsdbr = 0.4 * rmsdbr + 0.6 * oldrmsdbr;
+
+ //only update when values appear to be different
+ if(olddbl == dbl && olddbr == dbr)
+ return;
+
+ olddbl = dbl;
+ olddbr = dbr;
+ oldrmsdbl = rmsdbl;
+ oldrmsdbr = rmsdbr;
+
+ damage(FL_DAMAGE_USER1);
+ }
+ private:
+ float olddbl,olddbr;
+ float oldrmsdbl,oldrmsdbr;
+ float dbl,dbr,rmsdbl,rmsdbr,maxdbl,maxdbr;
+ int clipped;
+
+ Fl_Osc_Interface *osc;
+};
diff --git a/src/UI/VuMeter.h b/src/UI/VuMeter.h
@@ -0,0 +1,23 @@
+#ifndef VU_METER_H
+#define VU_METER_H
+#include <FL/Fl_Box.H>
+
+class VuMeter: public Fl_Box
+{
+ public:
+
+ VuMeter(int x,int y, int w, int h, const char *label=0)
+ :Fl_Box(x,y,w,h,label)
+ {}
+
+ protected:
+ float limit(float x)
+ {
+ if(x<0.0)
+ x=0.0;
+ else if(x>1.0)
+ x=1.0;
+ return x;
+ }
+};
+#endif
diff --git a/src/UI/VuPartMeter.h b/src/UI/VuPartMeter.h
@@ -0,0 +1,53 @@
+#include "VuMeter.h"
+#define MIN_DB (-48)
+
+class VuPartMeter: public VuMeter
+{
+ public:
+ VuPartMeter(int x,int y, int w, int h, const char *label=0)
+ :VuMeter(x,y,w,h,label)
+ {}
+
+ void draw(void)
+ {
+ const int X = x(), Y = y(), W = w(), H = h();
+
+ //XXX perhaps re-enable this later on
+ //if (!active_r()){
+ // int fakedb=master->fakepeakpart[npart];
+ // fl_rectf(X,Y,W,H,140,140,140);
+ // if (fakedb>0){
+ // fakedb=(int)(fakedb/255.0*H)+4;
+ // fl_rectf(X+2,Y+H-fakedb,W-4,fakedb,0,0,0);
+ // }
+ // return;
+ //}
+
+ //draw the vu lines
+ const int idb = db*(H-2);
+
+ fl_rectf(X,Y+H-idb,W,idb,0,200,255);
+ fl_rectf(X,Y,W,H-idb,0,0,0);
+
+ //draw the scales
+ const float tmp=H*1.0/MIN_DB;
+ for (int i = 1; i < 1 - MIN_DB; i++) {
+ const int ty = H+(int) (tmp*i);
+ if(i%5 == 0) fl_rectf(X, Y+H-ty, W, 1,0, 160, 200);
+ if(i%10 == 0) fl_rectf(X, Y+H-ty, W, 1,0, 230, 240);
+ }
+ }
+
+ void update(float x)
+ {
+ const float _db = limit((MIN_DB-rap2dB(x))/MIN_DB);
+
+ if(db != _db) {
+ db = _db;
+ damage(FL_DAMAGE_USER1);
+ }
+ }
+
+ private:
+ float db;
+};
diff --git a/src/main.cpp b/src/main.cpp
@@ -20,9 +20,6 @@
*/
-#include <FL/Fl.H>
-
-#include "UI/common.H"
#include <iostream>
#include <cmath>
@@ -46,29 +43,17 @@
#include "Misc/Dump.h"
extern Dump dump;
-
//Nio System
#include "Nio/Nio.h"
-#ifndef DISABLE_GUI
-#ifdef QT_GUI
-
-#include <QApplication>
-#include "masterui.h"
-QApplication *app;
+//GUI System
+#include "UI/Fl_Osc_Interface.h"
+#include "UI/Connection.h"
+GUI::ui_handle_t gui;
-#elif defined FLTK_GUI
-#include "UI/MasterUI.h"
-#elif defined NTK_GUI
-#include "UI/MasterUI.h"
-#include <FL/Fl_Shared_Image.H>
-#include <FL/Fl_Tiled_Image.H>
-#include <FL/Fl_Dial.H>
-#endif // FLTK_GUI
-
-MasterUI *ui;
-
-#endif //DISABLE_GUI
+//The Glue
+rtosc::ThreadLink *bToU = new rtosc::ThreadLink(1024,1024);
+rtosc::ThreadLink *uToB = new rtosc::ThreadLink(1024,1024);
using namespace std;
@@ -106,8 +91,6 @@ void error_cb(int i, const char *m, const char *loc)
}
lo_server server;
-rtosc::ThreadLink *bToU = new rtosc::ThreadLink(1024,1024);
-rtosc::ThreadLink *uToB = new rtosc::ThreadLink(1024,1024);
string last_url, curr_url;
void path_search(const char *m)
@@ -196,14 +179,16 @@ void osc_setup(void)
*/
void osc_check(void)
{
- lo_server_recv_noblock(server, 100);
+ lo_server_recv_noblock(server, 0);
while(bToU->hasNext()) {
const char *rtmsg = bToU->read();
if(!strcmp(rtmsg, "/echo")
&& !strcmp(rtosc_argument_string(rtmsg),"ss")
&& !strcmp(rtosc_argument(rtmsg,0).s, "OSC_URL"))
curr_url = rtosc_argument(rtmsg,1).s;
- else {
+ else if(curr_url == "GUI") {
+ GUI::raiseUi(gui, bToU->read());
+ } else{
lo_message msg = lo_message_deserialise((void*)rtmsg,
rtosc_message_length(rtmsg, bToU->buffer_size()), NULL);
@@ -217,28 +202,6 @@ void osc_check(void)
}
-#ifndef DISABLE_GUI
-
-#ifdef NTK_GUI
-static Fl_Tiled_Image *module_backdrop;
-#endif
-
-void
-set_module_parameters ( Fl_Widget *o )
-{
-#ifdef NTK_GUI
- o->box( FL_DOWN_FRAME );
- o->align( o->align() | FL_ALIGN_IMAGE_BACKDROP );
- o->color( FL_BLACK );
- o->image( module_backdrop );
- o->labeltype( FL_SHADOW_LABEL );
-#else
- o->box( FL_PLASTIC_UP_BOX );
- o->color( FL_CYAN );
- o->labeltype( FL_EMBOSSED_LABEL );
-#endif
-}
-#endif
/*
* Program initialisation
@@ -263,6 +226,21 @@ void initprogram(void)
osc_setup();
}
+class UI_Interface:public Fl_Osc_Interface
+{
+ void requestValue(string s)
+ {
+ if(last_url != "GUI") {
+ uToB->write("/echo", "ss", "OSC_URL", "GUI");
+ last_url = "GUI";
+ }
+
+ uToB->write(s.c_str(),"");
+
+ }
+} ui_link;
+
+
/*
* Program exit
*/
@@ -274,9 +252,7 @@ void exitprogram()
Nio::stop();
-#ifndef DISABLE_GUI
- delete ui;
-#endif
+ GUI::destroyUi(gui);
#if LASH
if(lash)
delete lash;
@@ -555,47 +531,16 @@ int main(int argc, char *argv[])
}
-#ifndef DISABLE_GUI
-
-#ifdef NTK_GUI
- fl_register_images();
-
- Fl_Dial::default_style(Fl_Dial::PIXMAP_DIAL);
-
- if(Fl_Shared_Image *img = Fl_Shared_Image::get(PIXMAP_PATH "/knob.png"))
- Fl_Dial::default_image(img);
- else
- Fl_Dial::default_image(Fl_Shared_Image::get(SOURCE_DIR "/../pixmaps/knob.png"));
-
- if(Fl_Shared_Image *img = Fl_Shared_Image::get(PIXMAP_PATH "/window_backdrop.png"))
- Fl::scheme_bg(new Fl_Tiled_Image(img));
- else
- Fl::scheme_bg(new Fl_Tiled_Image(Fl_Shared_Image::get(SOURCE_DIR "/../pixmaps/window_backdrop.png")));
-
- if(Fl_Shared_Image *img = Fl_Shared_Image::get(PIXMAP_PATH "/module_backdrop.png"))
- module_backdrop = new Fl_Tiled_Image(img);
- else
- module_backdrop = new Fl_Tiled_Image(Fl_Shared_Image::get(SOURCE_DIR "/../pixmaps/module_backdrop.png"));
+ gui = GUI::createUi(&ui_link, master, &Pexitprogram);
- Fl::background( 50, 50, 50 );
- Fl::background2( 70, 70, 70 );
- Fl::foreground( 255,255,255 );
-#endif
-
- ui = new MasterUI(master, &Pexitprogram);
-
- if ( !noui)
+ if(!noui)
{
- ui->showUI();
-
+ GUI::raiseUi(gui, "/show", "T");
if(!ioGood)
- fl_alert(
- "Default IO did not initialize.\nDefaulting to NULL backend.");
+ GUI::raiseUi(gui, "/alert", "s",
+ "Default IO did not initialize.\nDefaulting to NULL backend.");
}
-#endif
-
-#ifndef DISABLE_GUI
#if USE_NSM
char *nsm_url = getenv("NSM_URL");
@@ -610,7 +555,6 @@ int main(int argc, char *argv[])
}
}
#endif
-#endif
#if USE_NSM
if(!nsm)
@@ -618,17 +562,11 @@ int main(int argc, char *argv[])
{
#if LASH
lash = new LASHClient(&argc, &argv);
-#ifndef DISABLE_GUI
- ui->sm_indicator1->value(1);
- ui->sm_indicator2->value(1);
- ui->sm_indicator1->tooltip("LASH");
- ui->sm_indicator2->tooltip("LASH");
-#endif
+ GUI::raiseUi(gui, "/session-type", "s", "LASH");
#endif
}
while(Pexitprogram == 0) {
-#ifndef DISABLE_GUI
#if USE_NSM
if(nsm) {
nsm->check();
@@ -640,11 +578,11 @@ int main(int argc, char *argv[])
string filename;
switch(lash->checkevents(filename)) {
case LASHClient::Save:
- ui->do_save_master(filename.c_str());
+ GUI::raiseUi(gui, "/save-master", "s", filename.c_str());
lash->confirmevent(LASHClient::Save);
break;
case LASHClient::Restore:
- ui->do_load_master(filename.c_str());
+ GUI::raiseUi(gui, "/load-master", "s", filename.c_str());
lash->confirmevent(LASHClient::Restore);
break;
case LASHClient::Quit:
@@ -658,13 +596,8 @@ int main(int argc, char *argv[])
#if USE_NSM
done:
#endif
-
osc_check();
-
- Fl::wait(0.02f);
-#else
- usleep(100000);
-#endif
+ GUI::tickUi(gui);
}
exitprogram();