Merge CV, ver=5.1; ops/methods from HV, and interface from CV where possible
[goodguy/history.git] / cinelerra-5.1 / plugins / bluebanana / bluebananawindow.C
diff --git a/cinelerra-5.1/plugins/bluebanana/bluebananawindow.C b/cinelerra-5.1/plugins/bluebanana/bluebananawindow.C
new file mode 100644 (file)
index 0000000..fa6ac42
--- /dev/null
@@ -0,0 +1,2457 @@
+/*
+ * Cinelerra :: Blue Banana - color modification plugin for Cinelerra-CV
+ * Copyright (C) 2012 Monty <monty@xiph.org>
+ *
+ * This program is free software; you can redistribute it and/or modify
+ * it under the terms of the GNU General Public License as published by
+ * the Free Software Foundation; either version 2 of the License, or
+ * (at your option) any later version.
+ *
+ * This program is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+ * GNU General Public License for more details.
+ *
+ * You should have received a copy of the GNU General Public License
+ * along with this program; if not, write to the Free Software
+ * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA
+ *
+ */
+
+#include <unistd.h>
+#include <math.h>
+#include "bcdisplayinfo.h"
+#include "bcsignals.h"
+#include "cursors.h"
+#include "bluebanana.h"
+#include "bluebananaconfig.h"
+#include "bluebananaslider.h"
+#include "bluebananawindow.h"
+#include "keys.h"
+#include "language.h"
+#include "brender.h"
+
+#include "bluebananacolor.c"
+
+
+class BluebananaHActive : public BC_CheckBox {
+public:
+  BluebananaHActive(BluebananaMain *plugin, BluebananaWindow *gui);
+  virtual int handle_event();
+  void update();
+  BluebananaMain *plugin;
+  BluebananaWindow *gui;
+};
+
+class BluebananaSActive : public BC_CheckBox {
+public:
+  BluebananaSActive(BluebananaMain *plugin, BluebananaWindow *gui);
+  virtual int handle_event();
+  void update();
+  BluebananaMain *plugin;
+  BluebananaWindow *gui;
+};
+
+class BluebananaVActive : public BC_CheckBox {
+public:
+  BluebananaVActive(BluebananaMain *plugin, BluebananaWindow *gui);
+  virtual int handle_event();
+  void update();
+  BluebananaMain *plugin;
+  BluebananaWindow *gui;
+};
+
+
+// -------------------------------------------------------------------------------
+// The below code is because the stock TextTumbler trips too many bugs
+// in guicast to be useful. If this plugin is to be shipped standalone
+// and work, we need an independent version that avoids the bugs here.
+
+BB_Tumble::BB_Tumble(BluebananaMain *plugin,
+                     BluebananaWindow *gui,
+                     float min,
+                     float mid,
+                     float max,
+                     int precision,
+                     float increment,
+                     int text_w)
+  : BC_TextBox(-1,-1,text_w,1,mid,1,MEDIUMFONT,precision){
+
+  // We must pass in the precision for initialization to get the
+  // geometry right.  But then bctextbox.C immediately clobbers it, so
+  // set it again
+  set_precision(precision);
+
+  this->gui = gui;
+  this->plugin = plugin;
+  this->x = -1;
+  this->y = -1;
+  this->min = min;
+  this->mid = mid;
+  this->max = max;
+  this->text_w = text_w;
+  this->precision = precision;
+  this->increment = increment;
+  this->active = 0;
+  this->suppress_events = 0;
+}
+
+float BB_Tumble::get_value(){
+  const char *text = get_text();
+  if(!text || text[0]==0) return 0;
+  return atof(text);
+}
+
+int BB_Tumble::handle_event(){
+  if(!suppress_events) value_event();
+  return 1;
+}
+
+int BB_Tumble::activate(){
+  // we want to restore the previous value on ESC
+  prev = get_value();
+  // the textbox active member is private, make our own
+  active = 1;
+  return BC_TextBox::activate();
+}
+
+int BB_Tumble::deactivate(){
+  if(active){
+    // only fire an event if the value changed; this makes 0/360
+    // behavior in the hue selection readout stick with what the user
+    // actually sets (as opposed to sanity checking flipping the value
+    // if an event fires)
+    if(get_value()!=prev)
+      value_event();
+    active = 0;
+    suppress_events=0;
+  }
+  return BC_TextBox::deactivate();
+}
+
+int BB_Tumble::button_press_event(){
+  if(is_event_win()){
+    if(get_buttonpress() < 4) return BC_TextBox::button_press_event();
+    if(get_buttonpress() == 4){
+      float v = get_value()+increment;
+      if(v>max)v=max;
+      update(v);
+      value_event();
+    }else if(get_buttonpress() == 5){
+      float v = get_value()-increment;
+      if(v<min)v=min;
+      update(v);
+      value_event();
+    }
+    return 1;
+  }
+  return 0;
+}
+
+int BB_Tumble::keypress_event(){
+  if(active && get_keypress()==ESC)
+    update(prev);
+
+  // don't fire update events from every keypress when editing; that
+  // will trigger slider updates, which will then sanity-check/clamp
+  // values, and then update/clobber the readout text while we're
+  // editing it.
+  suppress_events=1;
+  int ret = BC_TextBox::keypress_event();
+  suppress_events=0;
+  return ret;
+}
+
+// --------------------------------------------------------------------------------
+//  initialization:
+//    1) create_objects() constructs sliders and readouts with
+//       limits but no values
+//    2) create_objects() then calls slider update()
+//       update computes and sets slider vals from config
+//       update computes and sets readout vals from config
+//
+//  slider drag event:
+//    1) slider calls handle_event(). This updates config.
+//    2) slider calls update()
+//       update computes and [re]sets slider vals from config;
+//          this is where the snap-to-value behavior comes from
+//       update computes and sets readout vals from config
+//
+//  readout event:
+//    1) readout calls handle_event(). This updates config.
+//    2) readout calls slider update() directly
+//       slider update computes and sets slider vals from config
+//       slider update computes and [re]sets readout vals from config;
+//           this does not result in a further handle_event() call, so
+//           no infinite recursion.
+//
+//  importantly, readout and slider values are set from the config
+//  (and vice-versa) in only one place each.
+
+// ---------------------------- hue adjustment slider -------------------------------
+
+class BluebananaHAReadout : public BB_Tumble {
+ public:
+  BluebananaHAReadout(BluebananaMain *plugin, BluebananaWindow *gui, int w)
+    : BB_Tumble(plugin,gui,rint(-180),0,rint(180), 0,1,w){}
+  int value_event();
+};
+
+class BluebananaHASlider : public BluebananaSliderSingle {
+public:
+  BluebananaHASlider(BluebananaMain *plugin, BluebananaWindow *gui,
+                     int x, int y, int w, int h)
+    : BluebananaSliderSingle(plugin,gui,x,y,w,h,-180,180){
+  }
+  virtual int handle_event() {
+    plugin->config.Hadj_val = val;
+    return 1;
+  }
+  void reset(){
+    plugin->config.Hadj_val=0;
+    update();
+  }
+  void update(){
+    val=plugin->config.Hadj_val;
+    highlight = plugin->config.active && plugin->config.Hadj_active;
+    gui->Hadj_readout->update(plugin->config.Hadj_val);
+    gui->slider_labels[7]->set_color(highlight && plugin->config.Hadj_val ?
+                                     get_resources()->default_text_color : dimtextcolor);
+    gui->enter_config_change();
+    gui->commit_config_change();
+  }
+  void trough_color(float hdel, float vdel, float &r, float &g, float &b, float &a){
+    float Hshift = plugin->config.Hsel_active ? (plugin->config.Hsel_lo + plugin->config.Hsel_hi)/720.f-.5f : 0.f;
+    float deg = hdel+Hshift;
+    if(deg<0)deg+=1;
+    if(deg>1)deg-=1;
+    HSpV_to_RGB(deg*6.f,1.,.2,r,g,b);
+    a=1.;
+  }
+};
+
+int BluebananaHAReadout::value_event(){
+  float val = get_value();
+  if(val<-180)val=-180;
+  if(val>180)val=180;
+  plugin->config.Hadj_val = val;
+  gui->Hadj_slider->update();
+  return 1;
+}
+
+// ------------------------------ Filter selection slider --------------------------------
+class BluebananaFSReadout0 : public BB_Tumble {
+ public:
+  BluebananaFSReadout0(BluebananaMain *plugin, BluebananaWindow *gui, int w)
+    : BB_Tumble(plugin,gui,-FSrange,0,FSrange, 0,1,w){}
+  int value_event();
+};
+class BluebananaFSReadout1 : public BB_Tumble {
+ public:
+  BluebananaFSReadout1(BluebananaMain *plugin, BluebananaWindow *gui, int w)
+    : BB_Tumble(plugin,gui,-FSrange,0,FSrange, 0,1,w){}
+  int value_event();
+};
+class BluebananaFSReadout2 : public BB_Tumble {
+ public:
+  BluebananaFSReadout2(BluebananaMain *plugin, BluebananaWindow *gui, int w)
+    : BB_Tumble(plugin,gui,-FSrange,0,FSrange, 0,1,w){}
+  int value_event();
+};
+class BluebananaFSReadout3 : public BB_Tumble {
+ public:
+  BluebananaFSReadout3(BluebananaMain *plugin, BluebananaWindow *gui, int w)
+    : BB_Tumble(plugin,gui,0.,0,FSovermax, 0,1,w){}
+  int value_event();
+};
+
+static void paint_dot(float *array,int w, int h, float cx, float cy, float r, int inv){
+  int x,y;
+  int x0 = floor(cx-r);
+  int x1 = ceil(cx+r)+1;
+  int y0 = floor(cy-r);
+  int y1 = ceil(cy+r)+1;
+
+  if(x0<0)x0=0;
+  if(x1<0)x1=0;
+  if(y0<0)y0=0;
+  if(y1<0)y1=0;
+
+  if(x0>w)x0=w;
+  if(x1>w)x1=w;
+  if(y0>h)y0=h;
+  if(y1>h)y1=h;
+
+  for(y=y0;y<y1;y++){
+    float *row = array+y*w;
+    for(x=x0;x<x1;x++){
+      float rr = hypotf(x-cx,y-cy);
+      /* not correct, but this is merely cosmetic anyway */
+      float v = (r-rr+.5);
+      if(v>0){
+        if(inv){
+          row[x] -= v;
+          if(row[x]>1.)row[x]=0.;
+        }else{
+          row[x] += v;
+          if(row[x]>1.)row[x]=1.;
+        }
+      }
+    }
+  }
+}
+
+class BluebananaFSSlider : public BluebananaSliderFill {
+public:
+  BluebananaFSSlider(BluebananaMain *plugin, BluebananaWindow *gui,
+                     int x, int y, int w, int h)
+    : BluebananaSliderFill(plugin,gui,x,y,w,h,-FSrange,FSrange,FSovermax) {
+    trough_alpha=0;
+    recompute_trough_alpha = 1;
+    erode=-1;
+  }
+  ~BluebananaFSSlider(){
+    if(trough_alpha)delete[] trough_alpha;
+  }
+  virtual int handle_event() {
+    plugin->config.Fsel_lo = (int)rint(loval);
+    plugin->config.Fsel_mid = (int)rint(midval);
+    plugin->config.Fsel_hi = (int)rint(hival);
+    plugin->config.Fsel_over = (int)rint(overval);
+    recompute_trough_alpha = 1;
+    return 1;
+  }
+  void update(){
+    if(plugin->config.Fsel_lo>0)plugin->config.Fsel_lo=0;
+    if(plugin->config.Fsel_hi<0)plugin->config.Fsel_hi=0;
+
+    if(highlight!=plugin->config.Fsel_active ||
+       erode!=plugin->config.Fsel_erode ||
+       loval!=plugin->config.Fsel_lo ||
+       midval!=plugin->config.Fsel_mid ||
+       hival!=plugin->config.Fsel_hi ||
+       overval!=plugin->config.Fsel_over){
+      recompute_trough_alpha = 1;
+    }
+    erode = plugin->config.Fsel_erode;
+    loval = plugin->config.Fsel_lo;
+    midval = plugin->config.Fsel_mid;
+    hival = plugin->config.Fsel_hi;
+    overval = plugin->config.Fsel_over;
+    highlight = plugin->config.Fsel_active;
+    gui->Fsel_readout0->update(plugin->config.Fsel_lo);
+    gui->Fsel_readout1->update(plugin->config.Fsel_mid);
+    gui->Fsel_readout2->update(plugin->config.Fsel_hi);
+    gui->Fsel_readout3->update(plugin->config.Fsel_over);
+    gui->slider_labels[3]->set_color
+      (highlight &&
+       (plugin->config.Hsel_active || plugin->config.Ssel_active || plugin->config.Vsel_active) &&
+       (plugin->config.Fsel_lo  != 0 ||
+        plugin->config.Fsel_mid != 0 ||
+        plugin->config.Fsel_hi  != 0 ||
+        plugin->config.Fsel_over!= 0 ) ?
+       get_resources()->default_text_color : dimtextcolor);
+
+    gui->erode_label->set_color
+      (highlight && plugin->config.Fsel_lo && plugin->config.Fsel_hi &&
+       (plugin->config.Hsel_active || plugin->config.Ssel_active || plugin->config.Vsel_active)?
+       get_resources()->default_text_color : dimtextcolor);
+
+    gui->enter_config_change();
+    gui->commit_config_change();
+  }
+  void trough_color(float hdel, float vdel, float &r, float &g, float &b, float &a){
+    int x = rint(hdel*troughcols-.5);
+    int y = rint(vdel*troughlines-.5);
+    float deg = plugin->config.Hsel_active ?
+      vdel*(plugin->config.Hsel_hi-plugin->config.Hsel_lo)+plugin->config.Hsel_lo :
+      vdel*360.f;
+    float sat = plugin->config.Ssel_active ?
+      (plugin->config.Ssel_hi+plugin->config.Ssel_lo)/200.:
+      .5;
+    float val = plugin->config.Vsel_active ?
+      (plugin->config.Vsel_hi*3+plugin->config.Vsel_lo)/400:
+      .75;
+
+    if(deg>=360)deg-=360.f;
+    HSpV_to_RGB(deg/60.,sat,val,r,g,b);
+    a = trough_alpha[troughcols*y+x];
+  }
+  void render(){
+
+    if(!trough_alpha)
+      trough_alpha = new float[troughcols*troughlines];
+
+    if(recompute_trough_alpha){
+      int trough_border = FSrange;
+      int tw = troughcols*3+trough_border*2;
+      int th = troughlines*3+trough_border*2;
+
+      float work1[tw*th];
+      float work2[tw*th];
+
+      memset(work1,0,sizeof(work1));
+
+      float loval=1;
+      float hival=FSrange*2-1;
+      float y0 = (th-1)/2.;
+      float y1 = (th-1)/2.;
+
+      int spacing=rint(FSrange)*2;
+      int rowflag=0;
+      int x,y;
+
+      while((y0+spacing*.5)>0){
+        for(x=(rowflag?spacing/2.:0.);x-spacing*.5<tw;x+=spacing){
+          float r = (((float)x/tw)*(hival-loval)+loval)*.5;
+          if(y0==y1){
+            paint_dot(work1,tw,th,x,y0,r,0);
+          }else{
+            paint_dot(work1,tw,th,x,y0,r,0);
+            paint_dot(work1,tw,th,x,y1,r,0);
+          }
+        }
+        y0-=spacing/2.;
+        y1+=spacing/2.;
+        rowflag = (rowflag+1)&1;
+      }
+
+      float *final = work1;
+      if(plugin->config.Fsel_active &&
+         (plugin->config.Fsel_lo || plugin->config.Fsel_hi || plugin->config.Fsel_over))
+        final=plugin->fill_selection(work1,work2,tw,th,NULL);
+
+      /* subsample into trough */
+      float *in = final + (tw+1)*trough_border;
+      for(y=0;y<troughlines;y++){
+        float *out = trough_alpha + troughcols*y;
+        for(x=0;x<troughcols;x++)
+          out[x] = (in[x*3]+in[x*3+1]+in[x*3+2]+
+                    in[tw+x*3]+in[tw+x*3+1]+in[x*3+2]+
+                    in[tw*2+x*3]+in[tw*2+x*3+1]+in[tw*2+x*3+2])*.1111;
+        in += tw*3;
+      }
+
+    }
+    recompute_trough_alpha=0;
+    BluebananaSliderFill::update();
+  }
+
+  float *trough_alpha;
+  int trough_border;
+  int recompute_trough_alpha;
+  int erode;
+};
+
+int BluebananaFSReadout0::value_event(){
+  float val = rint(get_value());
+  if(val<-FSrange)val=-FSrange;
+  if(val>0)val=0;
+  if(val>plugin->config.Fsel_mid) val = plugin->config.Fsel_mid;
+  plugin->config.Fsel_lo = val;
+  gui->Fsel_slider->update();
+  return 1;
+}
+
+int BluebananaFSReadout1::value_event(){
+  float val = rint(get_value());
+  if(val<-FSrange)val=-FSrange;
+  if(val>FSrange)val=FSrange;
+  if(val<plugin->config.Fsel_lo) plugin->config.Fsel_lo=val;
+  if(val>plugin->config.Fsel_hi) plugin->config.Fsel_hi=val;
+  plugin->config.Fsel_mid = val;
+  gui->Fsel_slider->update();
+  return 1;
+}
+
+int BluebananaFSReadout2::value_event(){
+  float val = rint(get_value());
+  if(val<0)val=0;
+  if(val>FSrange)val=FSrange;
+  if(val<plugin->config.Fsel_mid) val = plugin->config.Fsel_mid;
+  plugin->config.Fsel_hi = val;
+  gui->Fsel_slider->update();
+  return 1;
+}
+
+int BluebananaFSReadout3::value_event(){
+  float val = rint(get_value());
+  if(val<0)val=0;
+  if(val>FSovermax)val=FSovermax;
+  plugin->config.Fsel_over = val;
+  gui->Fsel_slider->update();
+  return 1;
+}
+
+// ------------------------------ value selection slider --------------------------------
+class BluebananaVSReadout0 : public BB_Tumble {
+ public:
+  BluebananaVSReadout0(BluebananaMain *plugin, BluebananaWindow *gui, int w)
+    : BB_Tumble(plugin,gui,0.,0,100., 0,1,w){}
+  int value_event();
+};
+class BluebananaVSReadout1 : public BB_Tumble {
+ public:
+  BluebananaVSReadout1(BluebananaMain *plugin, BluebananaWindow *gui, int w)
+    : BB_Tumble(plugin,gui,0.,0,100., 0,1,w){}
+  int value_event();
+};
+class BluebananaVSReadout2 : public BB_Tumble {
+ public:
+  BluebananaVSReadout2(BluebananaMain *plugin, BluebananaWindow *gui, int w)
+    : BB_Tumble(plugin,gui,0.,0,100., 0,1,w){}
+  int value_event();
+};
+
+class BluebananaVSSlider : public BluebananaSliderBracket {
+public:
+  BluebananaVSSlider(BluebananaMain *plugin, BluebananaWindow *gui,
+                     int x, int y, int w, int h)
+    : BluebananaSliderBracket(plugin,gui,x,y,w,h,0,100) { }
+  virtual int handle_event() {
+    plugin->config.Vsel_lo = rint(loval);
+    plugin->config.Vsel_hi = rint(hival);
+    plugin->config.Vsel_over = rint(overval);
+    return 1;
+  }
+  void pick(){
+    int delta = plugin->config.Vsel_hi - plugin->config.Vsel_lo;
+    float r = plugin->get_red();
+    float g = plugin->get_green();
+    float b = plugin->get_blue();
+    float h,s,v;
+    RGB_to_HSpV(r,g,b,h,s,v);
+    h*=60.f;
+    v=rint(v*100.f);
+    if(v<0)v=0;
+    if(v>100)v=100;
+    if(delta>25)delta=25;
+    int lo = v - delta/2;
+    int hi = lo + delta;
+    /* shrink the brackets if necessary */
+    if(lo<0){
+      lo=0;
+      if(hi-lo<10)hi=10;
+    }
+    if(hi>100){
+      hi=100;
+      if(hi-lo<10)lo=90;
+    }
+    plugin->config.Vsel_lo=lo;
+    plugin->config.Vsel_hi=hi;
+    plugin->config.Vsel_active=1;
+    gui->Vsel_active->update(); // this will also call the slider update
+  }
+  void update(){
+    loval = plugin->config.Vsel_lo;
+    hival = plugin->config.Vsel_hi;
+    midval = (loval+hival)/2.f;
+    overval = plugin->config.Vsel_over;
+    highlight = plugin->config.Vsel_active;
+
+    gui->Vsel_readout0->update(plugin->config.Vsel_lo);
+    gui->Vsel_readout1->update(plugin->config.Vsel_hi);
+    gui->Vsel_readout2->update(plugin->config.Vsel_over);
+    gui->slider_labels[2]->set_color
+      (highlight && (plugin->config.Vsel_lo != 0 || plugin->config.Vsel_hi != 100) ?
+       get_resources()->default_text_color : dimtextcolor);
+
+    gui->enter_config_change();
+    if(gui->Fsel_slider)gui->Fsel_slider->update();
+    gui->commit_config_change();
+  }
+  void trough_color(float hdel, float vdel, float &r, float &g, float &b, float &a){
+    float deg = plugin->config.Hsel_active ?
+      vdel*(plugin->config.Hsel_hi-plugin->config.Hsel_lo)+plugin->config.Hsel_lo :
+      vdel*360.f;
+    float sat = plugin->config.Ssel_active ?
+      (plugin->config.Ssel_hi*3+plugin->config.Ssel_lo)/400.:
+      .5;
+
+    if(deg>=360)deg-=360.f;
+    HSpV_to_RGB(deg/60.,sat,hdel,r,g,b);
+    a= plugin->val_select_alpha(hdel);
+  }
+};
+
+int BluebananaVSReadout0::value_event(){
+  float val = get_value();
+  if(val<0)val=0;
+  if(val>100)val=100;
+  if(val>plugin->config.Vsel_hi) val = plugin->config.Vsel_hi;
+  plugin->config.Vsel_lo = val;
+  gui->Vsel_slider->update();
+  return 1;
+}
+
+int BluebananaVSReadout1::value_event(){
+  float val = get_value();
+  if(val<0)val=0;
+  if(val>100)val=100;
+  if(val<plugin->config.Vsel_lo) val = plugin->config.Vsel_lo;
+  plugin->config.Vsel_hi = val;
+  gui->Vsel_slider->update();
+  return 1;
+}
+
+int BluebananaVSReadout2::value_event(){
+  float val = get_value();
+  if(val<0)val=0;
+  if(val>100)val=100;
+  plugin->config.Vsel_over = val;
+  gui->Vsel_slider->update();
+  return 1;
+}
+
+// ----------------------------- saturation selection slider -----------------------------
+class BluebananaSSReadout0 : public BB_Tumble {
+ public:
+  BluebananaSSReadout0(BluebananaMain *plugin, BluebananaWindow *gui, int w)
+    : BB_Tumble(plugin,gui,0.,0,100., 0,1,w){}
+  int value_event();
+};
+class BluebananaSSReadout1 : public BB_Tumble {
+ public:
+  BluebananaSSReadout1(BluebananaMain *plugin, BluebananaWindow *gui, int w)
+    : BB_Tumble(plugin,gui,0.,0,100., 0,1,w){}
+  int value_event();
+};
+class BluebananaSSReadout2 : public BB_Tumble {
+ public:
+  BluebananaSSReadout2(BluebananaMain *plugin, BluebananaWindow *gui, int w)
+    : BB_Tumble(plugin,gui,0.,0,100., 0,1,w){}
+  int value_event();
+};
+
+class BluebananaSSSlider : public BluebananaSliderBracket {
+public:
+  BluebananaSSSlider(BluebananaMain *plugin, BluebananaWindow *gui,
+                     int x, int y, int w, int h)
+    : BluebananaSliderBracket(plugin,gui,x,y,w,h,0,100) { }
+  int handle_event() {
+    plugin->config.Ssel_lo = rint(loval);
+    plugin->config.Ssel_hi = rint(hival);
+    plugin->config.Ssel_over = rint(overval);
+    return 1;
+  }
+  void pick(){
+    int delta = plugin->config.Ssel_hi - plugin->config.Ssel_lo;
+    float r = plugin->get_red();
+    float g = plugin->get_green();
+    float b = plugin->get_blue();
+    float h,s,v;
+    RGB_to_HSpV(r,g,b,h,s,v);
+    h*=60.f;
+    s=rint(s*100.f);
+    if(s<0)s=0;
+    if(s>100)s=100;
+    if(delta>25)delta=25;
+    int lo = s - delta/2;
+    int hi = lo + delta;
+    /* shrink the brackets if necessary */
+    if(lo<0){
+      lo=0;
+      if(hi-lo<10)hi=10;
+    }
+    if(hi>100){
+      hi=100;
+      if(hi-lo<10)lo=90;
+    }
+    plugin->config.Ssel_lo=lo;
+    plugin->config.Ssel_hi=hi;
+    plugin->config.Ssel_active=1;
+    gui->Ssel_active->update(); // this will also call the slider update
+  }
+  void update(){
+    loval = plugin->config.Ssel_lo;
+    hival = plugin->config.Ssel_hi;
+    midval = (loval+hival)/2.f;
+    overval = plugin->config.Ssel_over;
+    highlight = plugin->config.Ssel_active;
+
+    gui->Ssel_readout0->update(plugin->config.Ssel_lo);
+    gui->Ssel_readout1->update(plugin->config.Ssel_hi);
+    gui->Ssel_readout2->update(plugin->config.Ssel_over);
+    gui->slider_labels[1]->set_color(highlight  &&
+                                     (plugin->config.Ssel_lo != 0 || plugin->config.Ssel_hi != 100) ?
+                                     get_resources()->default_text_color : dimtextcolor);
+
+    gui->enter_config_change();
+    if(gui->Fsel_slider)gui->Fsel_slider->update();
+    gui->commit_config_change();
+  }
+  void trough_color(float hdel, float vdel, float &r, float &g, float &b, float &a){
+    float deg = plugin->config.Hsel_active ?
+      vdel*(plugin->config.Hsel_hi-plugin->config.Hsel_lo)+plugin->config.Hsel_lo :
+      vdel*360.;
+    if(deg>=360)deg-=360.f;
+    HSpV_to_RGB(deg/60.f,hdel,.7+.3*hdel,r,g,b);
+    a = plugin->sat_select_alpha(hdel);
+  }
+};
+
+int BluebananaSSReadout0::value_event(){
+  float val = get_value();
+  if(val<0)val=0;
+  if(val>100)val=100;
+  if(val>plugin->config.Ssel_hi) val = plugin->config.Ssel_hi;
+  plugin->config.Ssel_lo = val;
+  gui->Ssel_slider->update();
+  return 1;
+}
+
+int BluebananaSSReadout1::value_event(){
+  float val = get_value();
+  if(val<0)val=0;
+  if(val>100)val=100;
+  if(val<plugin->config.Ssel_lo) val = plugin->config.Ssel_lo;
+  plugin->config.Ssel_hi = val;
+  gui->Ssel_slider->update();
+  return 1;
+}
+
+int BluebananaSSReadout2::value_event(){
+  float val = get_value();
+  if(val<0)val=0;
+  if(val>100)val=100;
+  plugin->config.Ssel_over = val;
+  gui->Ssel_slider->update();
+  return 1;
+}
+
+// ----------------------------- hue selection slider ---------------------------------
+class BluebananaHSReadout0 : public BB_Tumble {
+ public:
+  BluebananaHSReadout0(BluebananaMain *plugin, BluebananaWindow *gui, int w)
+    : BB_Tumble(plugin,gui,0.,0,360., 0,1,w){}
+  int value_event();
+};
+class BluebananaHSReadout1 : public BB_Tumble {
+ public:
+  BluebananaHSReadout1(BluebananaMain *plugin, BluebananaWindow *gui, int w)
+    : BB_Tumble(plugin,gui,0.,0,360., 0,1,w){}
+  int value_event();
+};
+class BluebananaHSReadout2 : public BB_Tumble {
+ public:
+  BluebananaHSReadout2(BluebananaMain *plugin, BluebananaWindow *gui, int w)
+    : BB_Tumble(plugin,gui,0.,360.,360., 0,1,w){}
+  int value_event();
+};
+
+class BluebananaHSSlider : public BluebananaSliderCircular {
+public:
+  BluebananaHSSlider(BluebananaMain *plugin, BluebananaWindow *gui,
+                     int x, int y, int w, int h)
+    : BluebananaSliderCircular(plugin,gui,x,y,w,h,0,360) {
+  }
+  int handle_event() {
+    plugin->config.Hsel_lo = rint(loval);
+    plugin->config.Hsel_hi = plugin->config.Hsel_lo +
+      (midval<loval ? rint(midval*2-loval*2+720) : rint(midval*2-loval*2));
+    plugin->config.Hsel_over = rint(overval);
+    return 1;
+  }
+  void pick(){
+    int delta = plugin->config.Hsel_hi - plugin->config.Hsel_lo;
+    float r = plugin->get_red();
+    float g = plugin->get_green();
+    float b = plugin->get_blue();
+    float h,s,v;
+    RGB_to_HSpV(r,g,b,h,s,v);
+    h=rint(h*60.f);
+    if(h<0)h=0;
+    if(h>360)h=360;
+    if(delta>30)delta=30;
+    int lo = h - delta/2;
+    int hi = lo + delta;
+    if(lo<0){
+      lo+=360;
+      hi+=360;
+    }
+    plugin->config.Hsel_lo=lo;
+    plugin->config.Hsel_hi=hi;
+    plugin->config.Hsel_active=1;
+    gui->Hsel_active->update(); // this will also call the slider update
+  }
+  void update(){
+    // Significantly more complex than the other sliders due to the
+    // circular scale.
+    //float delta = (plugin->config.Hsel_hi - plugin->config.Hsel_lo);
+    loval = plugin->config.Hsel_lo;
+    overval = plugin->config.Hsel_over;
+    float newhival = plugin->config.Hsel_hi;
+    float newmidval = (loval+newhival)/2.;
+    if(loval<0)loval+=360.;
+    if(loval>360.)loval-=360.;
+    if(newmidval<0)newmidval+=360.;
+    if(newmidval>360.)newmidval-=360.;
+    if(newhival<0)newhival+=360.;
+    if(newhival>360.)newhival-=360.;
+    highlight = plugin->config.Hsel_active;
+
+    float checkhi = plugin->config.Hsel_hi;
+    if(checkhi>360)checkhi-=360;
+
+    // one last weirdness; 0 and 360 are technically the same value on
+    // the circular scale, but the user can set either.  This isn't a
+    // problem with lo as it doesn't wrap, but mid/hi is computed for
+    // the slider and could end up at 0 or 360, and then clobber the
+    // slider/readout.  To avoid annoying the user, don't override an
+    // existing readout/slider setting with its equivalent.
+    if(newhival==0 && hival==360.){
+      newhival=360.;
+      checkhi=360;
+    }else if(newhival==360 && hival==0.){
+      newhival=0.;
+      checkhi=0;
+    }else if(checkhi==0 && gui->Hsel_readout2->get_value()==360){
+      newhival=360.;
+      checkhi=360;
+    }else if(checkhi==360 && gui->Hsel_readout2->get_value()==0){
+      newhival=0.;
+      checkhi=0;
+    }
+
+    if(newmidval<1 && midval>359){
+      newmidval=360.;
+    }else if(newmidval>359.f && midval<1.){
+      newmidval=0.;
+    }
+    midval=newmidval;
+    hival=newhival;
+
+    gui->Hsel_readout0->update(plugin->config.Hsel_lo);
+    gui->Hsel_readout1->update(checkhi);
+    gui->Hsel_readout2->update(plugin->config.Hsel_over);
+    gui->slider_labels[0]->set_color(highlight &&
+                                     (plugin->config.Hsel_hi - plugin->config.Hsel_lo != 360) ?
+                                     get_resources()->default_text_color : dimtextcolor);
+
+    gui->enter_config_change();
+    if(gui->Fsel_slider)gui->Fsel_slider->update();
+    gui->commit_config_change();
+  }
+  void trough_color(float hdel, float vdel, float &r, float &g, float &b, float &a){
+    float deg = hdel*360.f;
+    if(deg>=360)deg-=360.f;
+
+    HSpV_to_RGB(deg/60.f,1.f,1.f,r,g,b);
+    a = plugin->hue_select_alpha(hdel*360.f);
+  }
+  friend class BluebananaHSReadout1;
+  friend class BluebananaHSReadout2;
+};
+
+int BluebananaHSReadout0::value_event(){
+  plugin->config.Hsel_lo = get_value();
+  if(plugin->config.Hsel_lo<0)plugin->config.Hsel_lo=0;
+  if(plugin->config.Hsel_lo>360)plugin->config.Hsel_lo=360;
+  if(plugin->config.Hsel_hi - plugin->config.Hsel_lo > 360)
+    plugin->config.Hsel_hi-=360;
+  if(plugin->config.Hsel_lo > plugin->config.Hsel_hi)
+    plugin->config.Hsel_hi+=360;
+  gui->Hsel_slider->update();
+  return 1;
+}
+
+int BluebananaHSReadout1::value_event(){
+  plugin->config.Hsel_hi = get_value();
+  if(plugin->config.Hsel_hi<0)plugin->config.Hsel_hi=0;
+  if(plugin->config.Hsel_hi>360)plugin->config.Hsel_hi=360;
+  if(plugin->config.Hsel_lo > plugin->config.Hsel_hi)
+    plugin->config.Hsel_hi+=360;
+  gui->Hsel_slider->hival=-1; /* force update to hival, not the hi readout */
+  gui->Hsel_slider->update();
+  return 1;
+}
+
+int BluebananaHSReadout2::value_event(){
+  float val = get_value();
+  if(val<0)val=0;
+  if(val>360)val=360;
+  plugin->config.Hsel_over=val;
+  gui->Hsel_slider->update();
+  return 1;
+}
+
+// ---------------------------- saturation adjustment slider ----------------------------
+class BluebananaSAReadout0 : public BB_Tumble {
+ public:
+  BluebananaSAReadout0(BluebananaMain *plugin, BluebananaWindow *gui, int w)
+    : BB_Tumble(plugin,gui,-100,0,100., 0,1,w){}
+  int value_event();
+};
+class BluebananaSAReadout1 : public BB_Tumble {
+ public:
+  BluebananaSAReadout1(BluebananaMain *plugin, BluebananaWindow *gui, int w)
+    : BB_Tumble(plugin,gui,0.,0,200., 0,1,w){}
+  int value_event();
+};
+class BluebananaSAReadout2 : public BB_Tumble {
+ public:
+  BluebananaSAReadout2(BluebananaMain *plugin, BluebananaWindow *gui, int w)
+    : BB_Tumble(plugin,gui,MIN_GAMMA,0,MAX_GAMMA,2,.01,w){}
+  int value_event();
+};
+
+class BluebananaSASlider : public BluebananaSliderChannel {
+public:
+  BluebananaSASlider(BluebananaMain *plugin, BluebananaWindow *gui,
+                     int x, int y, int w, int h)
+    : BluebananaSliderChannel(plugin,gui,x,y,w,h) { }
+  virtual int handle_event() {
+    plugin->config.Sadj_lo = loval;
+    plugin->config.Sadj_hi = hival;
+    plugin->config.Sadj_gamma = gamma;
+    return 1;
+  }
+  void reset(){
+    plugin->config.Sadj_lo=0.;
+    plugin->config.Sadj_hi=100.;
+    plugin->config.Sadj_gamma=1.;
+    update();
+  }
+  void update(){
+    loval = plugin->config.Sadj_lo;
+    hival = plugin->config.Sadj_hi;
+    gamma = plugin->config.Sadj_gamma;
+
+    highlight = plugin->config.active && plugin->config.Sadj_active;
+
+    gui->Sadj_readout0->update(plugin->config.Sadj_lo);
+    gui->Sadj_readout1->update(plugin->config.Sadj_hi);
+    gui->Sadj_readout2->update(plugin->config.Sadj_gamma);
+    gui->slider_labels[8]->set_color(highlight &&
+                                     (plugin->config.Sadj_lo != 0 ||
+                                      plugin->config.Sadj_hi != 100 ||
+                                      plugin->config.Sadj_gamma != 1.) ?
+                                     get_resources()->default_text_color : dimtextcolor);
+
+    gui->enter_config_change();
+    gui->commit_config_change();
+  }
+  void trough_color(float hdel, float &r, float &g, float &b){
+    r=g=b=0;
+  }
+};
+
+#define BBCLAMP(x,y,z) ((x)<(y) ? (y) : (x)>(z) ? (z) : (x))
+
+int BluebananaSAReadout0::value_event(){
+  plugin->config.Sadj_lo = BBCLAMP(get_value(),-100,plugin->config.Sadj_hi);
+  plugin->config.Sadj_lo = BBCLAMP(get_value(),-100,100);
+  gui->Sadj_slider->update();
+  return 1;
+}
+int BluebananaSAReadout1::value_event(){
+  plugin->config.Sadj_hi = BBCLAMP(get_value(),plugin->config.Sadj_lo,200);
+  plugin->config.Sadj_hi = BBCLAMP(get_value(),0,200);
+  gui->Sadj_slider->update();
+  return 1;
+}
+int BluebananaSAReadout2::value_event(){
+  plugin->config.Sadj_gamma = BBCLAMP(get_value(),MIN_GAMMA,MAX_GAMMA);
+  gui->Sadj_slider->update();
+  return 1;
+}
+
+// ------------------------------- value adjustment slider -------------------------------
+class BluebananaVAReadout0 : public BB_Tumble {
+ public:
+  BluebananaVAReadout0(BluebananaMain *plugin, BluebananaWindow *gui, int w)
+    : BB_Tumble(plugin,gui,-100.,0,100., 0,1,w){}
+  int value_event();
+};
+class BluebananaVAReadout1 : public BB_Tumble {
+ public:
+  BluebananaVAReadout1(BluebananaMain *plugin, BluebananaWindow *gui, int w)
+    : BB_Tumble(plugin,gui,0.,0,200., 0,1,w){}
+  int value_event();
+};
+class BluebananaVAReadout2 : public BB_Tumble {
+ public:
+  BluebananaVAReadout2(BluebananaMain *plugin, BluebananaWindow *gui, int w)
+    : BB_Tumble(plugin,gui,MIN_GAMMA,0,MAX_GAMMA,2,.01,w){}
+  int value_event();
+};
+
+class BluebananaVASlider : public BluebananaSliderChannel {
+public:
+  BluebananaVASlider(BluebananaMain *plugin, BluebananaWindow *gui,
+                     int x, int y, int w, int h)
+    : BluebananaSliderChannel(plugin,gui,x,y,w,h) { }
+  virtual int handle_event() {
+    plugin->config.Vadj_lo = loval;
+    plugin->config.Vadj_hi = hival;
+    plugin->config.Vadj_gamma = gamma;
+    return 1;
+  }
+  void reset(){
+    plugin->config.Vadj_lo=0;
+    plugin->config.Vadj_hi=100;
+    plugin->config.Vadj_gamma=1;
+    update();
+  }
+  void update(){
+    loval = plugin->config.Vadj_lo;
+    hival = plugin->config.Vadj_hi;
+    gamma = plugin->config.Vadj_gamma;
+
+    highlight = plugin->config.active && plugin->config.Vadj_active;
+
+    gui->Vadj_readout0->update(plugin->config.Vadj_lo);
+    gui->Vadj_readout1->update(plugin->config.Vadj_hi);
+    gui->Vadj_readout2->update(plugin->config.Vadj_gamma);
+    gui->slider_labels[9]->set_color(highlight  &&
+                                     (plugin->config.Vadj_lo != 0.f ||
+                                      plugin->config.Vadj_hi != 100.f ||
+                                      plugin->config.Vadj_gamma != 1.f) ?
+                                     get_resources()->default_text_color : dimtextcolor);
+
+    gui->enter_config_change();
+    gui->commit_config_change();
+  }
+  void trough_color(float hdel, float &r, float &g, float &b){
+    r=g=b=0;
+  }
+};
+
+int BluebananaVAReadout0::value_event(){
+  plugin->config.Vadj_lo = BBCLAMP(get_value(),-100,plugin->config.Vadj_hi);
+  plugin->config.Vadj_lo = BBCLAMP(get_value(),-100,100);
+  gui->Vadj_slider->update();
+  return 1;
+}
+int BluebananaVAReadout1::value_event(){
+  plugin->config.Vadj_hi = BBCLAMP(get_value(),plugin->config.Vadj_lo,200);
+  plugin->config.Vadj_hi = BBCLAMP(get_value(),0,200);
+  gui->Vadj_slider->update();
+  return 1;
+}
+int BluebananaVAReadout2::value_event(){
+  plugin->config.Vadj_gamma = BBCLAMP(get_value(),MIN_GAMMA,MAX_GAMMA);
+  gui->Vadj_slider->update();
+  return 1;
+}
+
+// -------------------------------- red adjustment slider --------------------------------
+class BluebananaRAReadout0 : public BB_Tumble {
+ public:
+  BluebananaRAReadout0(BluebananaMain *plugin, BluebananaWindow *gui, int w)
+    : BB_Tumble(plugin,gui,-100.,0,100., 0,1,w){}
+  int value_event();
+};
+class BluebananaRAReadout1 : public BB_Tumble {
+ public:
+  BluebananaRAReadout1(BluebananaMain *plugin, BluebananaWindow *gui, int w)
+    : BB_Tumble(plugin,gui,0.,0,200., 0,1,w){}
+  int value_event();
+};
+class BluebananaRAReadout2 : public BB_Tumble {
+ public:
+  BluebananaRAReadout2(BluebananaMain *plugin, BluebananaWindow *gui, int w)
+    : BB_Tumble(plugin,gui,MIN_GAMMA,0,MAX_GAMMA,2,.01,w){}
+  int value_event();
+};
+
+class BluebananaRASlider : public BluebananaSliderChannel {
+public:
+  BluebananaRASlider(BluebananaMain *plugin, BluebananaWindow *gui,
+                     int x, int y, int w, int h)
+    : BluebananaSliderChannel(plugin,gui,x,y,w,h) { }
+  virtual int handle_event() {
+    plugin->config.Radj_lo = loval;
+    plugin->config.Radj_hi = hival;
+    plugin->config.Radj_gamma = gamma;
+    return 1;
+  }
+  void reset(){
+    plugin->config.Radj_lo=0;
+    plugin->config.Radj_hi=100;
+    plugin->config.Radj_gamma=1;
+    update();
+  }
+  void update(){
+    loval = plugin->config.Radj_lo;
+    hival = plugin->config.Radj_hi;
+    gamma = plugin->config.Radj_gamma;
+
+    highlight = plugin->config.active && plugin->config.Radj_active;
+
+    gui->Radj_readout0->update(plugin->config.Radj_lo);
+    gui->Radj_readout1->update(plugin->config.Radj_hi);
+    gui->Radj_readout2->update(plugin->config.Radj_gamma);
+    gui->slider_labels[4]->set_color(highlight  &&
+                                     (plugin->config.Radj_lo != 0 ||
+                                      plugin->config.Radj_hi != 100 ||
+                                      plugin->config.Radj_gamma != 1) ?
+                                     get_resources()->default_text_color : dimtextcolor);
+
+    gui->enter_config_change();
+    gui->commit_config_change();
+  }
+  void trough_color(float hdel, float &r, float &g, float &b){
+    if(hdel<0){
+      r=g=b=0.;
+    }else if(hdel<=1.f){
+      r=hdel;
+      g=b=0.;
+    }else{
+      r=1.;
+      g=b=0.;
+    }
+  }
+};
+
+int BluebananaRAReadout0::value_event(){
+  plugin->config.Radj_lo = BBCLAMP(get_value(),-100,plugin->config.Radj_hi);
+  plugin->config.Radj_lo = BBCLAMP(get_value(),-100,100);
+  gui->Radj_slider->update();
+  return 1;
+}
+int BluebananaRAReadout1::value_event(){
+  plugin->config.Radj_hi = BBCLAMP(get_value(),plugin->config.Radj_lo,200);
+  plugin->config.Radj_hi = BBCLAMP(get_value(),0,200);
+  gui->Radj_slider->update();
+  return 1;
+}
+int BluebananaRAReadout2::value_event(){
+  plugin->config.Radj_gamma = BBCLAMP(get_value(),MIN_GAMMA,MAX_GAMMA);
+  gui->Radj_slider->update();
+  return 1;
+}
+
+// ---------------------------- green adjustment slider ----------------------------
+class BluebananaGAReadout0 : public BB_Tumble {
+ public:
+  BluebananaGAReadout0(BluebananaMain *plugin, BluebananaWindow *gui, int w)
+    : BB_Tumble(plugin,gui,-100.,0,100., 0,1,w){}
+  int value_event();
+};
+class BluebananaGAReadout1 : public BB_Tumble {
+ public:
+  BluebananaGAReadout1(BluebananaMain *plugin, BluebananaWindow *gui, int w)
+    : BB_Tumble(plugin,gui,0.,0,200., 0,1,w){}
+  int value_event();
+};
+class BluebananaGAReadout2 : public BB_Tumble {
+ public:
+  BluebananaGAReadout2(BluebananaMain *plugin, BluebananaWindow *gui, int w)
+    : BB_Tumble(plugin,gui,MIN_GAMMA,0,MAX_GAMMA,2,.01,w){}
+  int value_event();
+};
+
+class BluebananaGASlider : public BluebananaSliderChannel {
+public:
+  BluebananaGASlider(BluebananaMain *plugin, BluebananaWindow *gui,
+                     int x, int y, int w, int h)
+    : BluebananaSliderChannel(plugin,gui,x,y,w,h) { }
+  virtual int handle_event() {
+    plugin->config.Gadj_lo = loval;
+    plugin->config.Gadj_hi = hival;
+    plugin->config.Gadj_gamma = gamma;
+    return 1;
+  }
+  void reset(){
+    plugin->config.Gadj_lo=0;
+    plugin->config.Gadj_hi=100;
+    plugin->config.Gadj_gamma=1;
+    update();
+  }
+  void update(){
+    loval = plugin->config.Gadj_lo;
+    hival = plugin->config.Gadj_hi;
+    gamma = plugin->config.Gadj_gamma;
+
+    highlight = plugin->config.active && plugin->config.Gadj_active;
+
+    gui->Gadj_readout0->update(plugin->config.Gadj_lo);
+    gui->Gadj_readout1->update(plugin->config.Gadj_hi);
+    gui->Gadj_readout2->update(plugin->config.Gadj_gamma);
+    gui->slider_labels[5]->set_color(highlight  &&
+                                     (plugin->config.Gadj_lo != 0 ||
+                                      plugin->config.Gadj_hi != 100 ||
+                                      plugin->config.Gadj_gamma != 1) ?
+                                     get_resources()->default_text_color : dimtextcolor);
+
+    gui->enter_config_change();
+    gui->commit_config_change();
+  }
+  void trough_color(float hdel, float &r, float &g, float &b){
+    if(hdel<0){
+      r=g=b=0.;
+    }else if(hdel<=1.f){
+      g=hdel;
+      r=b=0.;
+    }else{
+      g=1.;
+      r=b=0.;
+    }
+  }
+};
+
+int BluebananaGAReadout0::value_event(){
+  plugin->config.Gadj_lo = BBCLAMP(get_value(),-100,plugin->config.Gadj_hi);
+  plugin->config.Gadj_lo = BBCLAMP(get_value(),-100,100);
+  gui->Gadj_slider->update();
+  return 1;
+}
+int BluebananaGAReadout1::value_event(){
+  plugin->config.Gadj_hi = BBCLAMP(get_value(),plugin->config.Gadj_lo,200);
+  plugin->config.Gadj_hi = BBCLAMP(get_value(),0,200);
+  gui->Gadj_slider->update();
+  return 1;
+}
+int BluebananaGAReadout2::value_event(){
+  plugin->config.Gadj_gamma = BBCLAMP(get_value(),MIN_GAMMA,MAX_GAMMA);
+  gui->Gadj_slider->update();
+  return 1;
+}
+
+// ------------------------------- blue adjustment slider -------------------------------
+class BluebananaBAReadout0 : public BB_Tumble {
+ public:
+  BluebananaBAReadout0(BluebananaMain *plugin, BluebananaWindow *gui, int w)
+    : BB_Tumble(plugin,gui,-100.,0,100., 0,1,w){}
+  int value_event();
+};
+class BluebananaBAReadout1 : public BB_Tumble {
+ public:
+  BluebananaBAReadout1(BluebananaMain *plugin, BluebananaWindow *gui, int w)
+    : BB_Tumble(plugin,gui,0.,0,200., 0,1,w){}
+  int value_event();
+};
+class BluebananaBAReadout2 : public BB_Tumble {
+ public:
+  BluebananaBAReadout2(BluebananaMain *plugin, BluebananaWindow *gui, int w)
+    : BB_Tumble(plugin,gui,MIN_GAMMA,0,MAX_GAMMA,2,.01,w){}
+  int value_event();
+};
+
+class BluebananaBASlider : public BluebananaSliderChannel {
+public:
+  BluebananaBASlider(BluebananaMain *plugin, BluebananaWindow *gui,
+                     int x, int y, int w, int h)
+    : BluebananaSliderChannel(plugin,gui,x,y,w,h) { }
+  virtual int handle_event() {
+    plugin->config.Badj_lo = loval;
+    plugin->config.Badj_hi = hival;
+    plugin->config.Badj_gamma = gamma;
+    return 1;
+  }
+  void reset(){
+    plugin->config.Badj_lo=0;
+    plugin->config.Badj_hi=100;
+    plugin->config.Badj_gamma=1;
+    update();
+  }
+  void update(){
+    loval = plugin->config.Badj_lo;
+    hival = plugin->config.Badj_hi;
+    gamma = plugin->config.Badj_gamma;
+
+    highlight = plugin->config.active && plugin->config.Badj_active;
+
+    gui->Badj_readout0->update(plugin->config.Badj_lo);
+    gui->Badj_readout1->update(plugin->config.Badj_hi);
+    gui->Badj_readout2->update(plugin->config.Badj_gamma);
+    gui->slider_labels[6]->set_color(highlight  &&
+                                     (plugin->config.Badj_lo != 0 ||
+                                      plugin->config.Badj_hi != 100 ||
+                                      plugin->config.Badj_gamma != 1) ?
+                                     get_resources()->default_text_color : dimtextcolor);
+
+    gui->enter_config_change();
+    gui->commit_config_change();
+  }
+  void trough_color(float hdel, float &r, float &g, float &b){
+    if(hdel<0){
+      r=g=b=0.;
+    }else if(hdel<=1.f){
+      b=hdel;
+      g=r=0.;
+    }else{
+      b=1.;
+      g=r=0.;
+    }
+  }
+};
+
+int BluebananaBAReadout0::value_event(){
+  plugin->config.Badj_lo = BBCLAMP(get_value(),-100,plugin->config.Badj_hi);
+  plugin->config.Badj_lo = BBCLAMP(get_value(),-100,100);
+  gui->Badj_slider->update();
+  return 1;
+}
+int BluebananaBAReadout1::value_event(){
+  plugin->config.Badj_hi = BBCLAMP(get_value(),plugin->config.Badj_lo,200);
+  plugin->config.Badj_hi = BBCLAMP(get_value(),0,200);
+  gui->Badj_slider->update();
+  return 1;
+}
+int BluebananaBAReadout2::value_event(){
+  plugin->config.Badj_gamma = BBCLAMP(get_value(),MIN_GAMMA,MAX_GAMMA);
+  gui->Badj_slider->update();
+  return 1;
+}
+
+// ---------------------------------- opacity slider ---------------------------------
+class BluebananaOAReadout : public BB_Tumble {
+ public:
+  BluebananaOAReadout(BluebananaMain *plugin, BluebananaWindow *gui, int w)
+    : BB_Tumble(plugin,gui,0.,0,100., 0,1,w){}
+  int value_event();
+};
+
+class BluebananaOASlider : public BluebananaSliderSingle {
+public:
+  BluebananaOASlider(BluebananaMain *plugin, BluebananaWindow *gui,
+                     int x, int y, int w, int h)
+    : BluebananaSliderSingle(plugin,gui,x,y,w,h,0,100) { }
+  virtual int handle_event() {
+    plugin->config.Oadj_val = val;
+    return 1;
+  }
+  void reset(){
+    plugin->config.Oadj_val=100;
+    update();
+  }
+  void update(){
+    val = plugin->config.Oadj_val;
+    highlight = plugin->config.active && plugin->config.Oadj_active;
+    gui->Oadj_readout->update(plugin->config.Oadj_val);
+    gui->slider_labels[10]->set_color(highlight  &&
+                                      plugin->config.Oadj_val != 100 ?
+                                      get_resources()->default_text_color : dimtextcolor);
+    gui->enter_config_change();
+    gui->commit_config_change();
+  }
+  void trough_color(float hdel, float vdel, float &r, float &g, float &b, float &a){
+    r=g=b=.8;
+    a=1-cos(hdel*M_PI*.5);
+  }
+};
+
+int BluebananaOAReadout::value_event(){
+  float val = get_value();
+  plugin->config.Oadj_val = val;
+  gui->Oadj_slider->update();
+  return 1;
+}
+
+// ------------------------------------- picker buttons -----------------------------------------
+class BluebananaHPicker : public BC_GenericButton{
+ public:
+  BluebananaHPicker(BluebananaWindow *gui, int w) : BC_GenericButton(-1, -1, w, _("Pick")){
+    this->gui = gui;
+  }
+  int handle_event() { gui->Hsel_slider->pick(); return 1; }
+  BluebananaWindow *gui;
+};
+class BluebananaSPicker : public BC_GenericButton{
+ public:
+  BluebananaSPicker(BluebananaWindow *gui, int w) : BC_GenericButton(-1, -1, w, _("Pick")){
+    this->gui = gui;
+  }
+  int handle_event() { gui->Ssel_slider->pick(); return 1; }
+  BluebananaWindow *gui;
+};
+class BluebananaVPicker : public BC_GenericButton{
+ public:
+  BluebananaVPicker(BluebananaWindow *gui, int w) : BC_GenericButton(-1, -1, w, _("Pick")){
+    this->gui = gui;
+  }
+  int handle_event() { gui->Vsel_slider->pick(); return 1; }
+  BluebananaWindow *gui;
+};
+
+// -------------------------------------- reset buttons -----------------------------------------
+
+class BluebananaHAReset : public BC_GenericButton{
+ public:
+  BluebananaHAReset(BluebananaWindow *gui, int w) : BC_GenericButton(-1, -1, w, _("Reset")){
+    this->gui = gui;
+  }
+  int handle_event() { gui->Hadj_slider->reset(); return 1;}
+  BluebananaWindow *gui;
+};
+class BluebananaSAReset : public BC_GenericButton{
+ public:
+  BluebananaSAReset(BluebananaWindow *gui, int w) : BC_GenericButton(-1, -1, w, _("Reset")){
+    this->gui = gui;
+  }
+  int handle_event() { gui->Sadj_slider->reset(); return 1;}
+  BluebananaWindow *gui;
+};
+class BluebananaVAReset : public BC_GenericButton{
+ public:
+  BluebananaVAReset(BluebananaWindow *gui, int w) : BC_GenericButton(-1, -1, w, _("Reset")){
+    this->gui = gui;
+  }
+  int handle_event() { gui->Vadj_slider->reset(); return 1;}
+  BluebananaWindow *gui;
+};
+class BluebananaRAReset : public BC_GenericButton{
+ public:
+  BluebananaRAReset(BluebananaWindow *gui, int w) : BC_GenericButton(-1, -1, w, _("Reset")){
+    this->gui = gui;
+  }
+  int handle_event() { gui->Radj_slider->reset(); return 1;}
+  BluebananaWindow *gui;
+};
+class BluebananaGAReset : public BC_GenericButton{
+ public:
+  BluebananaGAReset(BluebananaWindow *gui, int w) : BC_GenericButton(-1, -1, w, _("Reset")){
+    this->gui = gui;
+  }
+  int handle_event() { gui->Gadj_slider->reset(); return 1;}
+  BluebananaWindow *gui;
+};
+class BluebananaBAReset : public BC_GenericButton{
+ public:
+  BluebananaBAReset(BluebananaWindow *gui, int w) : BC_GenericButton(-1, -1, w, _("Reset")){
+    this->gui = gui;
+  }
+  int handle_event() { gui->Badj_slider->reset(); return 1;}
+  BluebananaWindow *gui;
+};
+class BluebananaOAReset : public BC_GenericButton{
+ public:
+  BluebananaOAReset(BluebananaWindow *gui, int w) : BC_GenericButton(-1, -1, w, _("Reset")){
+    this->gui = gui;
+  }
+  int handle_event() { gui->Oadj_slider->reset(); return 1;}
+  BluebananaWindow *gui;
+};
+
+// ----------------------------------- slider active buttons ------------------------------------
+
+BluebananaHActive::BluebananaHActive(BluebananaMain *plugin, BluebananaWindow *gui)
+  : BC_CheckBox(-1, -1, &plugin->config.Hsel_active, ""){
+  this->plugin = plugin;
+  this->gui = gui;
+}
+int BluebananaHActive::handle_event(){
+  plugin->config.Hsel_active = get_value();
+  update();
+  return 1;
+}
+void BluebananaHActive::update(){
+  this->BC_CheckBox::update(plugin->config.Hsel_active,1);
+  gui->Hsel_slider->update();
+}
+
+BluebananaSActive::BluebananaSActive(BluebananaMain *plugin, BluebananaWindow *gui)
+  : BC_CheckBox(-1, -1, &plugin->config.Ssel_active, ""){
+  this->plugin = plugin;
+  this->gui = gui;
+}
+int BluebananaSActive::handle_event(){
+  plugin->config.Ssel_active = get_value();
+  update();
+  return 1;
+}
+void BluebananaSActive::update(){
+  this->BC_CheckBox::update(plugin->config.Ssel_active,1);
+  gui->Ssel_slider->update();
+}
+
+BluebananaVActive::BluebananaVActive(BluebananaMain *plugin, BluebananaWindow *gui)
+  : BC_CheckBox(-1, -1, &plugin->config.Vsel_active, ""){
+  this->plugin = plugin;
+  this->gui = gui;
+}
+int BluebananaVActive::handle_event(){
+  plugin->config.Vsel_active = get_value();
+  update();
+  return 1;
+}
+void BluebananaVActive::update(){
+  this->BC_CheckBox::update(plugin->config.Vsel_active,1);
+  gui->Vsel_slider->update();
+}
+
+class BluebananaFActive : public BC_CheckBox {
+public:
+  BluebananaFActive(BluebananaMain *plugin, BluebananaWindow *gui)
+  : BC_CheckBox(-1, -1, &plugin->config.Fsel_active, ""){
+    this->plugin = plugin;
+    this->gui = gui;
+  }
+  virtual int handle_event(){
+    plugin->config.Fsel_active = get_value();
+    update();
+    return 1;
+  }
+  void update(){
+    this->BC_CheckBox::update(plugin->config.Fsel_active,1);
+    gui->Fsel_slider->update();
+  }
+  BluebananaMain *plugin;
+  BluebananaWindow *gui;
+};
+
+class BluebananaHAActive : public BC_CheckBox {
+public:
+  BluebananaHAActive(BluebananaMain *plugin, BluebananaWindow *gui)
+  : BC_CheckBox(-1, -1, &plugin->config.Hadj_active, ""){
+    this->plugin = plugin;
+    this->gui = gui;
+  }
+  virtual int handle_event(){
+    plugin->config.Hadj_active = get_value();
+    update();
+    return 1;
+  }
+  void update(){
+    this->BC_CheckBox::update(plugin->config.Hadj_active,1);
+    gui->Hadj_slider->update();
+  }
+  BluebananaMain *plugin;
+  BluebananaWindow *gui;
+};
+
+class BluebananaSAActive : public BC_CheckBox {
+public:
+  BluebananaSAActive(BluebananaMain *plugin, BluebananaWindow *gui)
+  : BC_CheckBox(-1, -1, &plugin->config.Sadj_active, ""){
+    this->plugin = plugin;
+    this->gui = gui;
+  }
+  virtual int handle_event(){
+    plugin->config.Sadj_active = get_value();
+    update();
+    return 1;
+  }
+  void update(){
+    this->BC_CheckBox::update(plugin->config.Sadj_active,1);
+    gui->Sadj_slider->update();
+  }
+  BluebananaMain *plugin;
+  BluebananaWindow *gui;
+};
+
+class BluebananaVAActive : public BC_CheckBox {
+public:
+  BluebananaVAActive(BluebananaMain *plugin, BluebananaWindow *gui)
+  : BC_CheckBox(-1, -1, &plugin->config.Vadj_active, ""){
+    this->plugin = plugin;
+    this->gui = gui;
+  }
+  virtual int handle_event(){
+    plugin->config.Vadj_active = get_value();
+    update();
+    return 1;
+  }
+  void update(){
+    this->BC_CheckBox::update(plugin->config.Vadj_active,1);
+    gui->Vadj_slider->update();
+  }
+  BluebananaMain *plugin;
+  BluebananaWindow *gui;
+};
+
+class BluebananaRAActive : public BC_CheckBox {
+public:
+  BluebananaRAActive(BluebananaMain *plugin, BluebananaWindow *gui)
+  : BC_CheckBox(-1, -1, &plugin->config.Radj_active, ""){
+    this->plugin = plugin;
+    this->gui = gui;
+  }
+  virtual int handle_event(){
+    plugin->config.Radj_active = get_value();
+    update();
+    return 1;
+  }
+  void update(){
+    this->BC_CheckBox::update(plugin->config.Radj_active,1);
+    gui->Radj_slider->update();
+  }
+  BluebananaMain *plugin;
+  BluebananaWindow *gui;
+};
+
+class BluebananaGAActive : public BC_CheckBox {
+public:
+  BluebananaGAActive(BluebananaMain *plugin, BluebananaWindow *gui)
+  : BC_CheckBox(-1, -1, &plugin->config.Gadj_active, ""){
+    this->plugin = plugin;
+    this->gui = gui;
+  }
+  virtual int handle_event(){
+    plugin->config.Gadj_active = get_value();
+    update();
+    return 1;
+  }
+  void update(){
+    this->BC_CheckBox::update(plugin->config.Gadj_active,1);
+    gui->Gadj_slider->update();
+  }
+  BluebananaMain *plugin;
+  BluebananaWindow *gui;
+};
+
+class BluebananaBAActive : public BC_CheckBox {
+public:
+  BluebananaBAActive(BluebananaMain *plugin, BluebananaWindow *gui)
+  : BC_CheckBox(-1, -1, &plugin->config.Badj_active, ""){
+    this->plugin = plugin;
+    this->gui = gui;
+  }
+  virtual int handle_event(){
+    plugin->config.Badj_active = get_value();
+    update();
+    return 1;
+  }
+  void update(){
+    this->BC_CheckBox::update(plugin->config.Badj_active,1);
+    gui->Badj_slider->update();
+  }
+  BluebananaMain *plugin;
+  BluebananaWindow *gui;
+};
+
+class BluebananaOAActive : public BC_CheckBox {
+public:
+  BluebananaOAActive(BluebananaMain *plugin, BluebananaWindow *gui)
+  : BC_CheckBox(-1, -1, &plugin->config.Oadj_active, ""){
+    this->plugin = plugin;
+    this->gui = gui;
+  }
+  virtual int handle_event(){
+    plugin->config.Oadj_active = get_value();
+    update();
+    return 1;
+  }
+  void update(){
+    this->BC_CheckBox::update(plugin->config.Oadj_active,1);
+    gui->Oadj_slider->update();
+  }
+  BluebananaMain *plugin;
+  BluebananaWindow *gui;
+};
+
+// -------------------------------------------- Erode --------------------------------------------
+class BluebananaErode : public BC_CheckBox {
+public:
+  BluebananaErode(BluebananaMain *plugin, BluebananaWindow *gui)
+  : BC_CheckBox(-1, -1, &plugin->config.Fsel_erode, ""){
+    this->plugin = plugin;
+    this->gui = gui;
+  }
+  virtual int handle_event(){
+    plugin->config.Fsel_erode = get_value();
+    update();
+    return 1;
+  }
+  void update(){
+    this->BC_CheckBox::update(plugin->config.Fsel_erode,1);
+    gui->Fsel_slider->update();
+  }
+  BluebananaMain *plugin;
+  BluebananaWindow *gui;
+};
+
+
+// -------------------------------------- Invert Selection ---------------------------------------
+class BluebananaIS : public BC_CheckBox {
+public:
+  BluebananaIS(BluebananaMain *plugin, BluebananaWindow *gui)
+  : BC_CheckBox(-1, -1, &plugin->config.invert_selection, ""){
+    this->plugin = plugin;
+    this->gui = gui;
+  }
+  virtual int handle_event(){
+    plugin->config.invert_selection = get_value();
+    update();
+    return 1;
+  }
+  void update(){
+    this->BC_CheckBox::update(plugin->config.invert_selection,1);
+    gui->enter_config_change();
+    gui->commit_config_change();
+  }
+  BluebananaMain *plugin;
+  BluebananaWindow *gui;
+};
+
+
+// -------------------------------------------- Mark --------------------------------------------
+class BluebananaMark : public BC_CheckBox {
+public:
+  BluebananaMark(BluebananaMain *plugin, BluebananaWindow *gui)
+    : BC_CheckBox(-1, -1, 0, ""){
+    this->plugin = plugin;
+    this->gui = gui;
+  }
+  virtual int handle_event() {
+    if(plugin->config.mark != get_value()){
+      plugin->config.mark = get_value();
+      plugin->save_nonauto();
+      if(plugin->config.mark){
+        gui->set_repeat(207);
+      }else{
+        gui->unset_repeat(207);
+      }
+      plugin->server->mwindow->sync_parameters();
+    }
+    return 1;
+  }
+  void update (){
+    if(plugin->config.mark != get_value()){
+      this->BC_CheckBox::update(plugin->config.mark,1);
+      if(plugin->config.mark){
+        gui->set_repeat(207);
+      }else{
+        gui->unset_repeat(207);
+      }
+    }
+  };
+  BluebananaMain *plugin;
+  BluebananaWindow *gui;
+};
+
+// ------------------------------------------- Active -------------------------------------------
+class BluebananaActive : public BC_CheckBox {
+public:
+  BluebananaActive(BluebananaMain *plugin, BluebananaWindow *gui)
+    : BC_CheckBox(-1, -1, &plugin->config.active, ""){
+    this->plugin = plugin;
+    this->gui = gui;
+    active=-1;
+  }
+  virtual int handle_event(){
+    active = get_value();
+    update();
+    return 1;
+  }
+  void update(){
+    if(active != plugin->config.active){
+      plugin->config.active = active;
+      this->BC_CheckBox::update(plugin->config.active,1);
+      gui->enter_config_change();
+      gui->Hadj_slider->update();
+      gui->Sadj_slider->update();
+      gui->Vadj_slider->update();
+      gui->Radj_slider->update();
+      gui->Gadj_slider->update();
+      gui->Badj_slider->update();
+      gui->Oadj_slider->update();
+      gui->commit_config_change();
+    }
+  }
+  BluebananaMain *plugin;
+  BluebananaWindow *gui;
+  int active;
+};
+
+// ---------------------------------------- Capture mask ---------------------------------------
+class BluebananaUnmask : public BC_CheckBox {
+public:
+  BluebananaUnmask(BluebananaMain *plugin, BluebananaWindow *gui,int padx)
+    : BC_CheckBox(-1, -1, &plugin->config.capture_mask, ""){
+    this->plugin = plugin;
+    this->gui = gui;
+    this->padx = padx;
+    this->label = new BC_Title(-1,-1,_(" End Mask"));
+    this->x=-1;
+    this->y=-1;
+    gui->add_subwindow(this->label);
+    gui->add_subwindow(this);
+    hidden = -1;
+  }
+  virtual int handle_event(){
+    plugin->config.capture_mask=get_value();
+    plugin->save_nonauto();
+    update();
+    gui->enter_config_change();
+    gui->commit_config_change();
+    return 1;
+  }
+  int get_h(){
+    return label->get_h();
+  }
+  int get_w(){
+    return BC_CheckBox::get_w()+label->get_w()+padx*4;
+  }
+  void reposition_window(int x, int y){
+    int h = label->get_h();
+    this->x = x;
+    this->y = y;
+    label->reposition_window(x+padx,y);
+    BC_CheckBox::reposition_window(x+padx*2+label->get_w(),y+(h-BC_CheckBox::get_h())/2);
+    update();
+  }
+  void update(){
+    int w = get_w();
+    int h = get_h();
+    int f=0;
+    int hideme = !plugin->config.use_mask;
+    switch(plugin->colormodel){
+    case BC_RGB888:
+    case BC_RGB_FLOAT:
+    case BC_YUV888:
+    case BC_RGB161616:
+    case BC_YUV161616:
+      hideme=1;
+      break;
+    }
+
+    if(hideme && hidden!=1){
+      hide_window();
+      label->hide_window();
+      gui->set_color(get_resources()->get_bg_color());
+      gui->draw_box(x,y,w,h);
+      gui->set_color(get_resources()->default_text_color);
+      gui->draw_line(x,y+h/2,x+w,y+h/2);
+      hidden=1;
+      f=1;
+    }
+
+    if(!hideme && hidden!=0){
+      gui->set_color(get_resources()->get_bg_color());
+      gui->draw_box(x,y,w,h);
+      show_window();
+      label->show_window();
+      hidden=0;
+      f=1;
+    }
+
+    if(plugin->config.capture_mask != get_value())
+      this->BC_CheckBox::update(plugin->config.capture_mask,1);
+    if(f)
+      gui->flash(x,y,w,h);
+  }
+  BluebananaMain *plugin;
+  BluebananaWindow *gui;
+  BC_Title *label;
+  int x,y,padx,hidden;
+};
+
+// ------------------------------------------ Use mask ----------------------------------------
+class BluebananaA2Sel : public BC_CheckBox {
+public:
+  BluebananaA2Sel(BluebananaMain *plugin, BluebananaWindow *gui,int padx)
+    : BC_CheckBox(-1, -1, &plugin->config.use_mask, ""){
+    this->plugin = plugin;
+    this->gui = gui;
+    this->padx = padx;
+    this->label = new BC_Title(-1,-1,_(" Mask Selection"));
+    this->x=-1;
+    this->y=-1;
+    gui->add_subwindow(this->label);
+    gui->add_subwindow(this);
+    hidden = -1;
+  }
+  virtual int handle_event(){
+    plugin->config.use_mask=get_value();
+    plugin->save_nonauto();
+    update();
+    gui->enter_config_change();
+    gui->commit_config_change();
+    return 1;
+  }
+  int get_h(){
+    return label->get_h();
+  }
+  int get_w(){
+    return BC_CheckBox::get_w()+label->get_w()+padx*4;
+  }
+  void reposition_window(int x, int y){
+    int h = label->get_h();
+    this->x = x;
+    this->y = y;
+    label->reposition_window(x+padx,y);
+    BC_CheckBox::reposition_window(x+padx*2+label->get_w(),y+(h-BC_CheckBox::get_h())/2);
+    update();
+  }
+  void update(){
+    int w = get_w();
+    int h = get_h();
+    int f=0;
+
+    if(gui->capture_mask)
+      gui->capture_mask->update();
+
+    switch(plugin->colormodel){
+    case BC_RGB888:
+    case BC_RGB_FLOAT:
+    case BC_YUV888:
+    case BC_RGB161616:
+    case BC_YUV161616:
+      if(hidden!=1){
+        hide_window();
+        label->hide_window();
+        gui->set_color(get_resources()->get_bg_color());
+        gui->draw_box(x,y,w,h);
+        gui->set_color(get_resources()->default_text_color);
+        gui->draw_line(x,y+h/2,x+w,y+h/2);
+        hidden=1;
+        f=1;
+      }
+      break;
+    case BC_RGBA8888:
+    case BC_RGBA_FLOAT:
+    case BC_YUVA8888:
+    case BC_RGBA16161616:
+    case BC_YUVA16161616:
+      if(hidden!=0){
+        gui->set_color(get_resources()->get_bg_color());
+        gui->draw_box(x,y,w,h);
+        show_window();
+        label->show_window();
+        hidden=0;
+        f=1;
+      }
+      break;
+    case -1:
+      // not initialized yet
+      return;
+    default:
+      fprintf(stderr,_("Unknown colormodel in BluebananaA2Sel:update()\n"));
+      break;
+    }
+    if(plugin->config.use_mask != get_value())
+      this->BC_CheckBox::update(plugin->config.use_mask,1);
+    if(f)
+      gui->flash(x,y,w,h);
+  }
+  BluebananaMain *plugin;
+  BluebananaWindow *gui;
+  BC_Title *label;
+  int x,y,padx,hidden;
+};
+
+// --------------------------------------- Main GUI window --------------------------------------
+BluebananaWindow::BluebananaWindow(BluebananaMain *plugin)
+ : PluginClientWindow(plugin,1000,1000,0,1,1)
+{
+  do_render=0;
+  windowx = get_x();
+  windowy = get_y();
+  this->plugin = plugin;
+  config_refcount=1; // suppress pushing config during startup
+  config_change=0;
+  config_produce=0;
+  config_consume=0;
+  config_pending=0;
+
+  Hsel_slider=NULL;
+  Ssel_slider=NULL;
+  Vsel_slider=NULL;
+  Fsel_slider=NULL;
+  Hadj_slider=NULL;
+  Sadj_slider=NULL;
+  Vadj_slider=NULL;
+  Radj_slider=NULL;
+  Gadj_slider=NULL;
+  Badj_slider=NULL;
+  Oadj_slider=NULL;
+
+  use_mask=0;
+  capture_mask=0;
+}
+
+BluebananaWindow::~BluebananaWindow()
+{
+}
+
+void BluebananaWindow::create_objects()
+{
+  int xmargin = 20, ymargin = 10;
+  float row_padding = .1;
+  float column_padding = .3;
+  int padx=0;
+
+  int i;
+  int row_h=0, row_adv=0;
+  int label_x=0;
+  int label_w=0;
+  int slider_w=0;
+  int picker_w=0;
+  int reset_w=0;
+  int tumbler_w=0,tumbler_ww=0,tumbler_h=0;
+  int tumbler_col1_x=0,tumbler_col2_x=0,tumbler_col2_w=0;
+  int y = ymargin;
+
+  //BluebananaHAReset *hareset=NULL;
+
+  config_refcount=0;
+  enter_config_change();
+
+  /* window headline */
+  {
+    BC_Title *l = new BC_Title(xmargin,y,_("Color Selection"));
+    BC_Title *l2 = new BC_Title(-1,-1,_(" Mark Selected Areas"));
+    add_subwindow(mark = new BluebananaMark(plugin,this));
+    add_subwindow(l);
+    add_subwindow(l2);
+    padx = l->get_h()*column_padding;
+    label_x = xmargin + l->get_w();
+
+    int x0 = get_w()-xmargin-mark->get_w();
+    mark->reposition_window(x0,y-(mark->get_h()-l->get_h())/2);
+    x0 -= padx+l2->get_w();
+    l2->reposition_window(x0,y);
+    x0-=padx;
+
+    set_color(get_resources()->default_text_color);
+    draw_line(label_x+padx, (int)(y+l->get_h()*.5), x0, (int)(y+l->get_h()*.5));
+
+    y += l->get_h()*(row_padding+1.);
+  }
+
+  const char *labels[11]={_("hue"),_("saturation"),_("value"),_("fill"),_("red"),_("green"),_("blue"),_("hue"),_("saturation"),_("value"),_("fade")};
+  for(i=0;i<11;i++){
+    add_subwindow(slider_labels[i] = new BC_Title(-1,-1,labels[i]));
+    if(slider_labels[i]->get_w()>label_w)label_w=slider_labels[i]->get_w();
+  }
+
+  int tumbler_text_ww = MAX(get_text_width(MEDIUMFONT,"-000"),get_text_width(MEDIUMFONT,"5.00"))+8;
+  int tumbler_text_w = get_text_width(MEDIUMFONT,"50")+8;
+  if(tumbler_text_w*3<tumbler_text_ww*2){
+    tumbler_text_ww = (tumbler_text_ww*2+2)/3*3/2;
+    tumbler_text_w=tumbler_text_ww*2/3;
+  }
+
+  erode_label = new BC_Title(xmargin,y,_("pre-erode"));
+  BluebananaErode *erode = new BluebananaErode(plugin,this);
+  add_subwindow(erode_label);
+  add_subwindow(erode);
+
+  for(i=0;i<11;i++){
+    BC_GenericButton *p=NULL;
+    BluebananaSlider *s=NULL;
+    BB_Tumble *t0 = NULL, *t1=NULL, *t2=NULL;
+    BC_Toggle *a = NULL;
+    BC_Title *l = slider_labels[i];
+
+    switch(i){
+    case 0:
+
+      add_subwindow(t0 = Hsel_readout0 = new BluebananaHSReadout0(plugin,this,tumbler_text_ww));
+      add_subwindow(t1 = Hsel_readout1 = new BluebananaHSReadout1(plugin,this,tumbler_text_ww));
+      add_subwindow(t2 = Hsel_readout2 = new BluebananaHSReadout2(plugin,this,tumbler_text_ww));
+      add_subwindow(a = Hsel_active = new BluebananaHActive(plugin,this));
+
+      /* need a narrow and a wide wide tumbler */
+      add_subwindow(Fsel_readout0 = new BluebananaFSReadout0(plugin,this,tumbler_text_w));
+      add_subwindow(Fsel_readout1 = new BluebananaFSReadout1(plugin,this,tumbler_text_w));
+      add_subwindow(Fsel_readout2 = new BluebananaFSReadout2(plugin,this,tumbler_text_w));
+      add_subwindow(Fsel_readout3 = new BluebananaFSReadout3(plugin,this,tumbler_text_ww));
+      tumbler_w = Fsel_readout0->get_w();
+      tumbler_ww = t0->get_w();
+      tumbler_h = t0->get_h();
+
+      /* need a reset button's width */
+      reset_w = BC_GenericButton::calculate_w(this, _("Reset"));
+      picker_w = BC_GenericButton::calculate_w(this, _("Pick"));
+
+      /* determine row spacing */
+      row_h = 30; /* minimum widget height allowance for the row
+                     (really, min height for the slider) */
+      if(row_h<a->get_h())row_h=a->get_h();
+      if(row_h<l->get_h())row_h=l->get_h();
+      if(row_h<BC_GenericButton::calculate_h())row_h=BC_GenericButton::calculate_h();
+      if(row_h<t2->get_h())row_h=t2->get_h();
+      row_adv = row_h*(1.+row_padding);
+
+      /* determine horizontal element positioning; two main setups */
+      /* setup 1: three tumblers + button */
+      tumbler_col2_w = MAX(reset_w,picker_w);
+
+      /* setup 2: four tumblers + erode */
+      {
+        int threew = tumbler_ww*3 + padx*4 + tumbler_col2_w;
+        int fourw = tumbler_w*3 + tumbler_ww + padx*5 + erode->get_w() + erode_label->get_w();
+        if(fourw>threew) tumbler_col2_w += fourw-threew;
+      }
+
+      tumbler_col2_x = get_w()-xmargin-tumbler_col2_w;
+      tumbler_col1_x = tumbler_col2_x - tumbler_ww*3 - padx*5;
+      slider_x = label_x+padx;
+      slider_w = (tumbler_col1_x - slider_x - padx*3);
+
+      /* make sure the label x doesn't cause any labels to go off the
+         left of the pane */
+      {
+        int lx = label_x - padx - a->get_w();
+        if (lx-label_w < xmargin){
+          slider_x += ((xmargin+label_w)-lx);
+          label_x += ((xmargin+label_w)-lx);
+          slider_w -= ((xmargin+label_w)-lx);
+        }
+      }
+
+      y += row_adv/3; /* extra half row spacing under headline */
+      s = Hsel_slider = new BluebananaHSSlider(plugin,this,slider_x,y,slider_w,row_h);
+      add_subwindow(p = new BluebananaHPicker(this,tumbler_col2_w));
+
+      /* Move the upper alpha <->selection config buttons into place */
+      {
+        int x0 = slider_x+slider_w+padx*2;
+        invert_selection = new BluebananaIS(plugin,this);
+        BC_Title *l = new BC_Title(xmargin,y,_(" Invert Selection"));
+        add_subwindow(l);
+        add_subwindow(invert_selection);
+        int w0 = padx+l->get_w()+padx+invert_selection->get_w()+padx*2;
+        set_color(get_resources()->get_bg_color());
+        draw_box(x0-w0,ymargin,w0,ymargin+l->get_h());
+        x0-=padx*2;
+        x0-=invert_selection->get_w();
+        invert_selection->reposition_window(x0,ymargin+(l->get_h()-invert_selection->get_h())/2);
+        x0-=padx;
+        x0-=l->get_w();
+        l->reposition_window(x0,ymargin);
+        x0-=padx;
+        x0-=padx*5;
+
+        use_mask = new BluebananaA2Sel(plugin,this,padx);
+        x0-=use_mask->get_w();
+        use_mask->reposition_window(x0, ymargin);
+        capture_mask = new BluebananaUnmask(plugin,this,padx);
+        x0-=padx*5 + capture_mask->get_w();
+        capture_mask->reposition_window(x0, ymargin);
+      }
+
+      break;
+
+    case 1:
+
+      add_subwindow(t0 = Ssel_readout0 = new BluebananaSSReadout0(plugin,this,tumbler_text_ww));
+      add_subwindow(t1 = Ssel_readout1 = new BluebananaSSReadout1(plugin,this,tumbler_text_ww));
+      add_subwindow(t2 = Ssel_readout2 = new BluebananaSSReadout2(plugin,this,tumbler_text_ww));
+      add_subwindow(a = Ssel_active = new BluebananaSActive(plugin,this));
+      add_subwindow(p = new BluebananaSPicker(this,tumbler_col2_w));
+      s = Ssel_slider = new BluebananaSSSlider(plugin,this,slider_x,y,slider_w,row_h);
+      break;
+
+    case 2:
+
+      add_subwindow(t0 = Vsel_readout0 = new BluebananaVSReadout0(plugin,this,tumbler_text_ww));
+      add_subwindow(t1 = Vsel_readout1 = new BluebananaVSReadout1(plugin,this,tumbler_text_ww));
+      add_subwindow(t2 = Vsel_readout2 = new BluebananaVSReadout2(plugin,this,tumbler_text_ww));
+      add_subwindow(a = Vsel_active = new BluebananaVActive(plugin,this));
+      add_subwindow(p = new BluebananaVPicker(this,tumbler_col2_w));
+      s = Vsel_slider = new BluebananaVSSlider(plugin,this,slider_x,y,slider_w,row_h);
+      break;
+
+    case 3:
+
+      add_subwindow(a = Fsel_active = new BluebananaFActive(plugin,this));
+      s = Fsel_slider = new BluebananaFSSlider(plugin,this,slider_x,y,slider_w,row_h);
+
+      Fsel_readout0->reposition_window(tumbler_col1_x, y + (row_h-tumbler_h)/2 + 1);
+      Fsel_readout1->reposition_window(tumbler_col1_x+tumbler_w, y + (row_h-tumbler_h)/2 + 1);
+      Fsel_readout2->reposition_window(tumbler_col1_x+tumbler_w*2, y + (row_h-tumbler_h)/2 + 1);
+      Fsel_readout3->reposition_window(tumbler_col1_x+tumbler_w*3+padx*2,
+                                       y + (row_h-tumbler_h)/2 + 1);
+
+      {
+        int x = get_w() - xmargin - erode->get_w();
+        erode->reposition_window(x,y+(row_h-erode->get_h())/2);
+        erode_label->reposition_window(x-erode_label->get_w()-padx,y+(row_h-erode_label->get_h())/2);
+      }
+
+      break;
+
+    case 4:
+
+      add_subwindow(t0 = Radj_readout0 = new BluebananaRAReadout0(plugin,this,tumbler_text_ww));
+      add_subwindow(t1 = Radj_readout1 = new BluebananaRAReadout1(plugin,this,tumbler_text_ww));
+      add_subwindow(t2 = Radj_readout2 = new BluebananaRAReadout2(plugin,this,tumbler_text_ww));
+      add_subwindow(a = Radj_active = new BluebananaRAActive(plugin,this));
+      add_subwindow(p = new BluebananaRAReset(this,tumbler_col2_w));
+      s = Radj_slider = new BluebananaRASlider(plugin,this,slider_x,y,slider_w,row_h);
+      break;
+
+    case 5:
+
+      add_subwindow(t0 = Gadj_readout0 = new BluebananaGAReadout0(plugin,this,tumbler_text_ww));
+      add_subwindow(t1 = Gadj_readout1 = new BluebananaGAReadout1(plugin,this,tumbler_text_ww));
+      add_subwindow(t2 = Gadj_readout2 = new BluebananaGAReadout2(plugin,this,tumbler_text_ww));
+      add_subwindow(a = Gadj_active = new BluebananaGAActive(plugin,this));
+      add_subwindow(p = new BluebananaGAReset(this,tumbler_col2_w));
+      s = Gadj_slider = new BluebananaGASlider(plugin,this,slider_x,y,slider_w,row_h);
+      break;
+
+    case 6:
+
+      add_subwindow(t0 = Badj_readout0 = new BluebananaBAReadout0(plugin,this,tumbler_text_ww));
+      add_subwindow(t1 = Badj_readout1 = new BluebananaBAReadout1(plugin,this,tumbler_text_ww));
+      add_subwindow(t2 = Badj_readout2 = new BluebananaBAReadout2(plugin,this,tumbler_text_ww));
+      add_subwindow(a = Badj_active = new BluebananaBAActive(plugin,this));
+      add_subwindow(p = new BluebananaBAReset(this,tumbler_col2_w));
+      s = Badj_slider = new BluebananaBASlider(plugin,this,slider_x,y,slider_w,row_h);
+      break;
+
+    case 7:
+
+      add_subwindow(t0 = Hadj_readout = new BluebananaHAReadout(plugin,this,tumbler_text_ww));
+      add_subwindow(a = Hadj_active = new BluebananaHAActive(plugin,this));
+      add_subwindow(p = new BluebananaHAReset(this,tumbler_col2_w));
+      s = Hadj_slider = new BluebananaHASlider(plugin,this,slider_x,y,slider_w,row_h);
+      break;
+
+    case 8:
+
+      add_subwindow(t0 = Sadj_readout0 = new BluebananaSAReadout0(plugin,this,tumbler_text_ww));
+      add_subwindow(t1 = Sadj_readout1 = new BluebananaSAReadout1(plugin,this,tumbler_text_ww));
+      add_subwindow(t2 = Sadj_readout2 = new BluebananaSAReadout2(plugin,this,tumbler_text_ww));
+
+      add_subwindow(a = Sadj_active = new BluebananaSAActive(plugin,this));
+      add_subwindow(p = new BluebananaSAReset(this,tumbler_col2_w));
+      s = Sadj_slider = new BluebananaSASlider(plugin,this,slider_x,y,slider_w,row_h);
+      break;
+
+    case 9:
+
+      add_subwindow(t0 = Vadj_readout0 = new BluebananaVAReadout0(plugin,this,tumbler_text_ww));
+      add_subwindow(t1 = Vadj_readout1 = new BluebananaVAReadout1(plugin,this,tumbler_text_ww));
+      add_subwindow(t2 = Vadj_readout2 = new BluebananaVAReadout2(plugin,this,tumbler_text_ww));
+      add_subwindow(a = Vadj_active = new BluebananaVAActive(plugin,this));
+      add_subwindow(p = new BluebananaVAReset(this,tumbler_col2_w));
+      s = Vadj_slider = new BluebananaVASlider(plugin,this,slider_x,y,slider_w,row_h);
+      break;
+
+    case 10:
+
+      add_subwindow(t0 = Oadj_readout = new BluebananaOAReadout(plugin,this,tumbler_text_ww));
+      add_subwindow(a = Oadj_active = new BluebananaOAActive(plugin,this));
+      add_subwindow(p = new BluebananaOAReset(this,tumbler_col2_w));
+      s = Oadj_slider = new BluebananaOASlider(plugin,this,slider_x,y,slider_w,row_h);
+      break;
+
+    }
+    add_subwindow(s);
+
+    if(a)a->reposition_window(label_x - padx - a->get_w(), y + (row_h-a->get_h())/2 + 1);
+    if(l)l->reposition(label_x - l->get_w() - padx*2 - a->get_w(), y + (row_h-l->get_h())/2 + 1);
+
+    if(p){
+      p->BC_SubWindow::reposition_window(tumbler_col2_x, y + (row_h-p->get_h())/2 + 1,
+                                         MAX(tumbler_col2_w,picker_w),p->get_h());
+
+      // work around bug; the reposition step does not fully redraw the button
+      p->draw_face();
+    }
+
+    if(t0)t0->reposition_window(tumbler_col1_x,
+                                y + (row_h-tumbler_h)/2 + 1);
+
+    if(t1)t1->reposition_window(tumbler_col1_x+tumbler_ww,
+                                y + (row_h-tumbler_h)/2 + 1);
+
+    if(t2)t2->reposition_window(tumbler_col1_x+tumbler_ww*2+padx*2+(tumbler_ww-t2->get_w())/2,
+                                y + (row_h-tumbler_h)/2 + 1);
+
+
+    s->update();
+
+    y+=row_adv;
+    if(i==3){
+      y+=row_adv/3;
+
+      BC_Title *l = new BC_Title(xmargin,y,_("Color Adjustment"));
+      BC_Title *l2 = new BC_Title(-1,-1,_(" Filter Active"));
+      add_subwindow(l);
+      add_subwindow(l2);
+      add_subwindow(active = new BluebananaActive(plugin,this));
+
+      int x0 = get_w()-xmargin-mark->get_w();
+      active->reposition_window(x0,y-(active->get_h()-l->get_h())/2);
+      x0 -= padx+l2->get_w();
+      l2->reposition_window(x0,y);
+      x0 -= padx*2;
+      set_color(get_resources()->default_text_color);
+      draw_line(xmargin+l->get_w()+padx, y+l->get_h()*.5, x0, y+l->get_h()*.5);
+
+      y += l->get_h()*(row_padding+1.);
+      y += row_adv/3;
+    }
+
+
+    if(i==6 || i==9){
+      y+=row_adv/4;
+
+      set_color((s->dimtextcolor + get_resources()->default_text_color)/2);
+      draw_line(slider_x+20, y+l->get_h()*.5, tumbler_col2_x-30, y+l->get_h()*.5);
+
+      y += l->get_h()*(row_padding+1.);
+      y += row_adv/4;
+    }
+
+
+  }
+  y += row_adv/2-4;
+
+  plugin->update_lookups(0);
+  do_render=1;
+
+  resize_window(get_w(),y);
+  show_window();
+  reposition_window(windowx,windowy,get_w(),y);
+  leave_config_change(); // also forces render
+  plugin->server->mwindow->sync_parameters();
+}
+
+int BluebananaWindow::close_event(){
+  set_done(1);
+  return 1;
+}
+
+// adds one to config push refcount
+// updates any internal state immediately
+void BluebananaWindow::enter_config_change(){
+  config_refcount++;
+  if(!config_change && !plugin->update_cache.equivalent(plugin->config)){
+    config_change=1;
+  }
+  plugin->update_lookups(0);
+}
+
+// decrements one from config push refcount.  If refcount drops to
+// zero, pushes new config up to application
+
+// also compresses events; waits 184ms for new events before pushing
+// configuration changes
+void BluebananaWindow::commit_config_change(){
+  if(--config_refcount==0){
+    if(config_change){
+      config_change=0;
+      config_produce++;
+      config_pending=1;
+      set_repeat(97);
+    }
+    render();
+  }
+}
+
+// decrements one from config push refcount.  Does not push config up
+// to application when refcount drops to zero (used to wrap update
+// requests coming from the application, not user-initiated state
+// changes)
+void BluebananaWindow::leave_config_change(){
+  config_change=0;
+  plugin->update_cache.copy_from(plugin->config);
+  if(--config_refcount==0){
+    render();
+  }
+}
+
+int BluebananaWindow::flush_config_change(){
+  unset_repeat(97);
+  if(config_pending){
+    config_pending=0;
+    plugin->update_cache.copy_from(plugin->config);
+    plugin->send_configure_change();
+  }
+  config_consume=config_produce;
+  return 0;
+}
+
+int BluebananaWindow::repeat_event(int64_t d){
+  if(d==97){
+    if(config_consume==config_produce)
+      flush_config_change();
+    config_consume=config_produce;
+  }
+  if(d==207){
+
+    /* if background render is active and we're showing the zebra, mark
+       the current frame uncached so that we can push zebra changes */
+    if(plugin->config.mark && plugin->server->mwindow->brender)
+      plugin->server->mwindow->brender->set_video_map(plugin->source_position, BRender::SCANNED);
+
+    /* push update request without an EDL update */
+    plugin->server->mwindow->sync_parameters();
+  }
+  return 0;
+}
+
+/* engine -> gui update; don't allow any EDL pushes */
+void BluebananaWindow::update(){
+
+  // called to suppress configuration pushes
+  enter_config_change();
+
+  // full configuration recompute and redraw
+  Hsel_slider->update();
+  Ssel_slider->update();
+  Vsel_slider->update();
+  Fsel_slider->update();
+  Hadj_slider->update();
+  Sadj_slider->update();
+  Vadj_slider->update();
+  Radj_slider->update();
+  Gadj_slider->update();
+  Badj_slider->update();
+  Oadj_slider->update();
+
+  active->update();
+  mark->update();
+  use_mask->update();
+  capture_mask->update();
+  invert_selection->update();
+
+  Hsel_active->update();
+  Ssel_active->update();
+  Vsel_active->update();
+  Fsel_active->update();
+
+  Hadj_active->update();
+  Sadj_active->update();
+  Vadj_active->update();
+  Radj_active->update();
+  Gadj_active->update();
+  Badj_active->update();
+  Oadj_active->update();
+
+  // called to release configuration without pushing
+  leave_config_change();
+}
+
+void BluebananaWindow::render(){
+  if(do_render){
+    Hsel_slider->render();
+    Ssel_slider->render();
+    Vsel_slider->render();
+    Fsel_slider->render();
+    Hadj_slider->render();
+    Sadj_slider->render();
+    Vadj_slider->render();
+    Radj_slider->render();
+    Gadj_slider->render();
+    Badj_slider->render();
+    Oadj_slider->render();
+  }
+}
+
+void BluebananaWindow::update_histograms(BluebananaMain *plugin){
+  int w = plugin->frame->get_w();
+  int h = plugin->frame->get_h();
+
+  if(Radj_slider)Radj_slider->update_histogram(plugin->red_histogram,0,0,0,w*h);
+  if(Gadj_slider)Gadj_slider->update_histogram(plugin->green_histogram,0,0,0,w*h);
+  if(Badj_slider)Badj_slider->update_histogram(plugin->blue_histogram,0,0,0,w*h);
+
+  if(Hadj_slider)
+    Hadj_slider->update_histogram(plugin->hue_histogram,
+                                  plugin->hue_histogram_red,
+                                  plugin->hue_histogram_green,
+                                  plugin->hue_histogram_blue,w*h);
+
+  if(Sadj_slider)
+    Sadj_slider->update_histogram(plugin->sat_histogram,
+                                  plugin->sat_histogram_red,
+                                  plugin->sat_histogram_green,
+                                  plugin->sat_histogram_blue,w*h);
+
+  if(Vadj_slider)
+    Vadj_slider->update_histogram(plugin->value_histogram,
+                                  plugin->value_histogram_red,
+                                  plugin->value_histogram_green,
+                                  plugin->value_histogram_blue,w*h);
+
+}