Subversion Repositories DIN Is Noise

Rev

Rev 1519 | Rev 1532 | Go to most recent revision | Blame | Compare with Previous | Last modification | View Log | RSS feed

/*
* ui.cc
* DIN Is Noise is copyright (c) 2006-2020 Jagannathan Sampath
* For more information, please visit https://dinisnoise.org/
*/



#include "ui_list.h"
#include "din.h"
#include "morse_code.h"
#include "console.h"
#include "curve_editor.h"
#include "tcl_interp.h"
#include "command.h"
#include "chrono.h"
#include "keyboard_keyboard.h"
#include "viewwin.h"
#include "authors_note.h"
#include "fft.h"
#include "main.h"
#include "curve_picker.h"
#include "capturer.h"
#include "fractaliser.h"
#include "warper.h"
#include "recorder.h"
#include "mondrian.h"
#include "binaural_drones.h"
#include "spiraler.h"
#include "circler.h"
#include "rose_milker.h"
#include "sine_mixer.h"
#include <fstream>
#include <string>
#include "container.h"

using namespace std;

extern string user_data_dir;
extern curve_editor waved, comed, octed, veled, drone_mod_ed, ran_mod_ed, ran_wh_ed, pitch_vol_ed, pomo_ed, delayed, drone_pend_ed;
extern int mousex, mousey, mouseyy;
extern string INSTRUMENT;
extern int rmb;
extern const int MILLION;
extern oscilloscope scope;
extern int line_height;
extern sine_mixer sinemixer;
extern const char* voice_is_lbls[];
extern const int NUM_INSTRUMENTS;

ui_list::ui_list () :
fed (fnt, "font.ed", "font.hlp"),
vlis (&cb_voice, &fdr_voice, &din0.dinfo.voice),
glis (&cb_gater, &fdr_gater, &din0.dinfo.gater),
dlis (&cb_delay, &fdr_delay, &din0.dinfo.delay)
{
  current = 0;
  prev = 0;
  crved = 0;
  lasted = 0;
  esct = -1;
  rmb_clicked = 0;
  LISTEN (ol_voice_is, &vivl);
}

void ui_list::set_current (ui* u) {

  if (current) {
    current->leave ();
    prev = current;
  }

  current = u;
  cons << YELLOW << "@ " << current->name << eol;

  if (u->ed) {
    crved = dynamic_cast<curve_editor*>(u);
    lasted = crved;
  } else uis[0] = current;

  current->enter ();

}

void ui_list::load_editor (ui* ed) {
  if (current->ed == 0) main_menu.setup_tabs (ed);
  set_current (ed);
}

void ui_list::bg () {

  current->calc_win_mouse ();

  for (vector<ui*>::size_type i = 1, j = uis.size(); i < j; ++i) uis[i]->bg (); // bg for instruments and editors

  // fade fx for voice, gater & delay
  eval_fade (fdr_voice, cb_voice);
  flash_gater (); // flashes in sync with gater value
  eval_fade (fdr_delay, cb_delay);

}

float ui_list::eval_fade (fader& fdr, checkbutton& cb) {
  if (fdr.eval ()) cb.blend_on_off_color (fdr.alpha);
  return fdr.amount;
}

void ui_list::flash_gater () {

  static color clr [2] = {color (1, 0, 0), color (0, 1, 0)};
  float dg = aout.gatr [aout.last_sample];
  if (fdr_gater.eval ()) {
    cb_gater.blend_on_off_color (fdr_gater.alpha);
    color& c = const_cast<color&> (cb_gater.clr);
    c *= dg;
  } else {
    int gi = din0.dinfo.gater;
    cb_gater.set_color (dg * clr [gi].r, dg * clr [gi].g, dg * clr[gi].b);
  }

}

int ui_list::is_widget_on_screen (widget* w, ui* scr) {
  vector<widget*>& widgets = widgets_of [scr];
  for (int i = 0, j = widgets.size (); i < j; ++i) if (w == widgets[i]) return 1;
  return 0;
}

int ui_list::handle_input () {

  // handle quit
  //
  #ifdef __MACOSX_CORE__
  if (keydown (SDLK_LMETA) && keypressed (SDLK_q)) { // command key + q  on mac os x
    quit = IMMEDIATE;
    return 1;
  }
  #else
  if (ALT && keypressed (SDLK_F4)) { // alt + f4 on windows & linux
    quit = IMMEDIATE;
    return 1;
  }
  #endif

 
  // handle widgets
  //
  basic_editor::hide_cursor = mouse_slider0.active;
  if (widget::focus) { // handle input of widget with focus
    widget::focus->handle_input ();
    return 1;
  } else {
    vector<widget*> widgets = widgets_of [current];
    for (vector<widget*>::size_type i = 0, j = widgets.size (); i < j; ++i) {
      widget* wi = widgets[i];
      if (wi->visible && wi->enabled) {
        wi->handle_input ();
        if (widget::focus) return 1;
      }
    }
  }

  // toggle menu
  //
  #ifdef __EVALUATION__
    if (current != &anote) { // no menu on author's note
  #endif
    if (rmb) {
      if (!rmb_clicked) {
        rmb_clicked = 1;
        if (escape_from_things ());
        else main_menu.toggle ();
      }
    } else rmb_clicked = 0;
  #ifdef __EVALUATION__
  }
  #endif

  // handle current screen
  current->handle_input ();

  if (keypressed (SDLK_1)) { // switch instrument
    if (current->ed == 0) goto_next_instrument (); // else back to current instrument
    load_instrument ();
  } else if (keypressed (SDLK_BACKQUOTE)) { // flip between instrument and last used editor
    if (current->ed)
      load_instrument ();
    else {
      if (lasted) load_editor (lasted);
    }
  } else { // load editors attached to keys 2 - 8
    for (int i = 0; i < MAX_EDITORS; ++i) {
      if (keypressed (key[i])) {
        ui* edi = ed[i];
        if (edi) load_editor (edi);
      }
    }
  }
 
  /*if (keypressed(SDLK_RETURN)) {
    if (alt_down ()) { // clear recording
      main_menu.recl.clicked (main_menu.b_clear_record);
    } else if (CTRL) { // start recording
      if (cb_record.state == 0) cb_record.turn_on (); else cb_record.turn_off ();
    }
  }*/


  if (keypressed (SDLK_F2)) {
    if (UI_OFF) turn_on_ui (); else turn_off_ui ();
  }

  if (keypressed (SDLK_PAUSE)) {
    scope.visible = !scope.visible;
    main_menu.cb_scope.set_state (scope.visible, 0);
  }

  if (keypressed (SDLK_MENU)) main_menu.mbl.clicked (main_menu.b_menu);

  if (keypressed (SDLK_ESCAPE)) {
    if (UI_OFF) turn_on_ui ();
    else if ( escape_from_things() ) ;
    else { // ask to press esc again to really quit
      esc:
      if (esct == -1) {
        cons << console::red << "Press ESC again to quit" << eol;
        esct = ui_clk();
      } else {
        double dt = ui_clk() - esct;
        if (dt > 1.0) { // 1 sec timeout
          esct = -1;
          goto esc;
        } else if (quit != TRY) try_quit ();
      }
    }
  }

  return 1;

}

int ui_list::set_editor (const string& name, int screen) {
  if (screen > 0) {
    ui* ei = 0;
    for (int i = 0, j = uis.size(); i < j; ++i) {
      if (uis[i]->name == name) {
        ei = uis[i];
        break;
      }
    }
    if (ei) {
      ed [screen - 2] = ei;
      return 1;
    } else {
      cmdlst.result = "bad editor name";
      return 0;
    }
  } else {
    cmdlst.result = "bad screen number (valid: 2 to 8)";
    return 0;
  }
}

void ui_list::draw () {
  glClear (GL_COLOR_BUFFER_BIT);
  glMatrixMode (GL_MODELVIEW);
  glLoadIdentity ();
  current->draw ();
  glMatrixMode (GL_PROJECTION);
  glLoadIdentity ();
  glOrtho (0, view.xmax, 0, view.ymax, 0, 1);
  vector<widget*> widgets = widgets_of [current];
  for (vector<widget*>::size_type i = 0, j = widgets.size (); i < j; ++i) {
    widget* wi = widgets[i];
    if (wi->visible) wi->draw ();
  }
}

// all screens
ui* SCREENS [] = {
  &keybd2,
  &mondrian0,
  &din0,
  &binaural_drones0,
  &din0.waved,
  &din0.droneed,
  &drone_mod_ed,
  &din0.moded,
  &din0.gated,
  &keybd2.waved,
  &keybd2.attacked,
  &keybd2.decayed,
  &keybd2.veled,
  &delayed,
  &octed,
  &comed,
  &mc.ed,
  &mondrian0.waved,
  &mondrian0.attacked,
  &mondrian0.decayed,
  &binaural_drones0.waved,
  &ran_mod_ed,
  &ran_wh_ed,
  &pitch_vol_ed,
  &pomo_ed,
  &noiser::ed,
  &drone_pend_ed,
  &uis.fed,
  &fractaliser_.ed,
  &warper_.ed,
  &spiraler_.scr.sin_ed,
  &spiraler_.scr.cos_ed,
  &spiraler_.scr.rad_ed,
  &rosemilker.scr.sin_ed,
  &rosemilker.scr.cos_ed,
  &circler_.scr.sin_ed,
  &circler_.scr.cos_ed,
  &circler_.scr.rad_ed,
  &sinemixer.sin_ed,
  &rosemilker.scr.rad_ed,
  0
};

void ui_list::show_hide_widgets (int sh) {
  for (ui** p = SCREENS; *p != 0; ++p) {
    ui* u = *p;
    vector<widget*>& vu = widgets_of [u];
    int j = vu.size ();
    if (j) {
      if (sh)
        for (int i = 0; i < j; ++i) vu[i]->show();
      else
        for (int i = 0; i < j; ++i) vu[i]->hide();
    }
  }
  curve_picker.hide ();
}

void ui_list::add_widgets () {

  widget* wall [] = {&cons, &main_menu, &b_settings}; // appears on all screens
  for (ui** p = SCREENS; *p != 0; ++p) for (int i = 0; i < 3; ++i) widgets_of[*p].push_back (wall[i]);

  // add selectors to microtonal keyboard and mondrian
  mkb_selector.set_listener (&din0);
  mon_selector.set_listener (&mondrian0);

  // add fft display on waveform editors
  ui* uwav [] = {&din0.waved, &keybd2.waved, &din0.droneed, &mondrian0.waved, &binaural_drones0.waved};
  for (int i = 0; i < 5; ++i) widgets_of[uwav[i]].push_back (&fft0);

  // add plugin browser to these editors
  ui* ueds [] = {
    &din0.waved,
    &din0.moded,
    &din0.droneed,
    &drone_mod_ed,
    &keybd2.waved,
    &fractaliser_.ed,
    &warper_.ed,
    &mondrian0.waved,
    &din0.gated,
    &binaural_drones0.waved,
    &circler_.scr.sin_ed,
    &circler_.scr.cos_ed,
    &sinemixer.sin_ed,
    &rosemilker.scr.sin_ed,
    &rosemilker.scr.cos_ed,
    &spiraler_.scr.sin_ed,
    &spiraler_.scr.cos_ed,
    &ran_mod_ed,
    &pitch_vol_ed,
    &pomo_ed,
    &drone_pend_ed
  };
  for (int i = 0; i < 21; ++i) widgets_of[ueds[i]].push_back (&plugin__browser);

  widget* dpeuw [] = {&dpeu.depth, &dpeu.bpm};
  widget_load ("d_dpeu", dpeuw, 2);
  dpeu.depth.set ("Depth", 1.0f, -MILLION, MILLION, &dpeul);
  dpeu.bpm.set ("BPM", 1.0f, -MILLION, MILLION, &dpeul);
  widgets_of [&drone_pend_ed].push_back (&dpeu.depth);
  widgets_of [&drone_pend_ed].push_back (&dpeu.bpm);

  /*MOVE (dpeu.depth);
  MOVE (dpeu.bpm);*/


  // add widgets for curve editors only
  for (ui** p = SCREENS + NUM_INSTRUMENTS; *p != 0; ++p) {
    ui* u = *p;
    vector<widget*>& wu = widgets_of [u];
    curve_editor* ce = dynamic_cast<curve_editor*>(u);
    wu.insert (wu.begin(), &curve_picker);
    wu.insert (wu.begin(), &ce->capturer);
    wu.insert (wu.begin(), &ce->pomo);
    ce->capturer.setup (ce);
    ce->pomo.setup (ce);
  }

  // add scope to all instruments
  extern instrument* INSTRUMENT_PTR[];
  for (int i = 0; i < NUM_INSTRUMENTS; ++i) widgets_of[(ui *)INSTRUMENT_PTR[i]].push_back (&scope);

  // no menu on author's note & settings screen
  widgets_of[&anote].push_back(&cons);
  widgets_of[&settings_scr].push_back (&cons);

  widget* wdin [] = {
    &ol_voice_is,
    &cb_voice,
    &cb_gater,
    &cb_delay,
    &cb_compress,
    &ab_scroll_left,
    &ab_scroll_right,
    &ab_scroll_up,
    &ab_scroll_down,
    &cb_show_pitch_volume_board,
    &cb_show_pitch_volume_drones,
    &cb_record,
    &din0.dinfo.gravity,
    &mkb_selector,
 };
 for (int i = 0, j = 14; i < j; ++i) widgets_of [&din0].push_back (wdin[i]);

  widget* wkeybd2 [] = {
    &cb_delay,
    &cb_compress,
    &sp_attack_time,
    &sp_decay_time,
    &sp_voices,
    &sp_pitch_bend,
    &cb_show_nearby_notes,
    &d_parameters,
    &ab_parameters,
    &l_waveform_display,
    &ab_prev_wav,
    &ab_next_wav,
    &cd_waveform_display,
    &l_octave_shift,
    &ab_octave_down,
    &ab_octave_up,
    &sp_octave_shift_bpm,
    &b_abort_octave_shift,
    &cb_record,
    &ol_trig_what
  };
 
  for (int i = 0, j = 20; i < j; ++i) widgets_of [&keybd2].push_back (wkeybd2[i]);

  widget* wmondrian [] = {
    &l_mondrian_voices,
    &cb_delay,
    &cb_compress,
    &cb_record,
    &mon_selector,
  };
  for (int i = 0, j = 5; i < j; ++i) widgets_of [&mondrian0].push_back (wmondrian[i]);
  l_mondrian_voices.set_text ("Voices: ** / **");

  widget* wsbd [] = {
    &cb_delay,
    &cb_compress,
    &cb_record,
  };

  for (int i = 0, j = 3; i < j; ++i) widgets_of [&binaural_drones0].push_back (wsbd[i]);


}

void ui_list::setup () {

  // all screens
  int nscreens = 40;
  uis.resize (nscreens);
  uis[0] = 0;
  ui** si = SCREENS;
  for (int i = 1; i < nscreens; ++i, ++si) uis[i] = *si;

  // bottom line
  int dirs [] = {arrow_button::left, arrow_button::right, arrow_button::up, arrow_button::down};
  arrow_button* scrl [] = {&ab_scroll_left, &ab_scroll_right, &ab_scroll_up, &ab_scroll_down};
  for (int i = 0; i < 4; ++i) {
    arrow_button* si = scrl[i];
    si->set_dir (dirs[i]);
    si->set_size (12);
    si->set_listener (&sal);
    si->click_repeat = 1;
    si->first_repeat_time = 0.005;
    si->subsequent_repeat_time = 0.015;
  }

  cb_voice.set_listener (&vlis);
  cb_voice.colorize (0);

  cb_gater.set_listener (&glis);
  cb_gater.colorize (0);

  cb_delay.set_listener (&dlis);
  cb_delay.colorize (0);

  cb_compress.set_listener (&clis);

  b_settings.set_listener (&slis);

  cb_record.set_listener (&main_menu.recl);

  sp_voices.set ("Voices", 1, 1, MILLION, &vov, 0);
  sp_attack_time.set ("Attack time", 0.1f, 0.0f, MILLION, &atv, 0);
  sp_decay_time.set ("Decay time", 0.1f, 0.0f, MILLION, &dkv, 0);
  sp_pitch_bend.set ("Hz/Pixel", 0.01f, 0.0f, MILLION, &pbl, 0);
  sp_octave_shift_bpm.set ("BPM", 1, 0.0f, MILLION, &osl, 0);

  cb_show_nearby_notes.set_listener (&pbl);
  cb_show_nearby_notes.turn_off ();

  // waveform display
  //
  l_waveform_display.set_text ("Waveform");

  ab_prev_wav.set_dir (arrow_button::left);
  ab_prev_wav.set_listener (&wdl);
  ab_next_wav.set_dir (arrow_button::right);
  ab_next_wav.set_listener (&wdl);

  ab_prev_wav.click_repeat = ab_next_wav.click_repeat = 1;
  ab_prev_wav.first_repeat_time = ab_next_wav.first_repeat_time = 0.33;
  ab_prev_wav.subsequent_repeat_time = ab_next_wav.subsequent_repeat_time = 1/20.;

  cd_waveform_display.set_size (96, 96);
  cd_waveform_display.crv = &keybd2.wave; // keyboard-keyboard's waveform
  cd_waveform_display.calc_bbox ();

  // octave shift
  l_octave_shift.set_text ("Octave Shift");
  ab_octave_down.set_dir (arrow_button::left);
  ab_octave_down.set_listener (&osl);
  ab_octave_up.set_dir (arrow_button::right);
  ab_octave_up.set_listener (&osl);
  int arrow_size = 24;
  ab_octave_up.set_size (arrow_size);
  ab_octave_down.set_size (arrow_size);
  b_abort_octave_shift.set_text ("Abort");
  b_abort_octave_shift.set_listener (&main_menu.aosl);

  // keys trigger? notes or noise
  ol_trig_what.set_listener (&twl);

  DEFINE_PARAMETERS
  widget_load ("d_parameters", pw, npars);
  for (int i = 1; i < npars; ++i) {
    d_parameters.add_child (pw[i]);
    //pw[i]->set_moveable(1);
  }

  d_parameters.set_name ("Parameters");
  d_parameters.set_moveable (1);

  ab_parameters.set_listener (&pal);

  cb_voice.set_state (din0.dinfo.voice);
  cb_gater.set_state (din0.dinfo.gater);
  cb_delay.set_state (din0.dinfo.delay);
  cb_compress.set_state (din0.dinfo.compress);

  cb_show_pitch_volume_board.set_listener (&spvl);
  cb_show_pitch_volume_drones.set_listener (&spvl);

#ifdef __EVALUATION__
  anote.setup ();
#endif

  plugin__browser.setup ();
  settings_scr.setup ();
  main_menu.setup ();

}

void ui_list::update_widgets (int wnow, int hnow, int wprev, int hprev) {

  main_menu.update ();

  plugin__browser.update ();

  cb_gater.set_text ("Gater");
  cb_delay.set_text ("Delay");
  cb_compress.set_text ("Compressor");
  b_settings.set_text ("Settings");
  cb_record.set_text ("Record");
  l_mondrian_voices.update ();
   
  int lh = line_height;
  int y = update_bottom_line ();
  y += lh;

  arrow_button* scrl [] = {&ab_scroll_left, &ab_scroll_right, &ab_scroll_up, &ab_scroll_down};
  int nscrls = 4;
  int w = ab_scroll_left.extents.width;
  int dw [] = {5, 7, 0, 0};
  int x = (view.xmax - nscrls * w) / 2;
  for (int i = 0; i < nscrls; ++i) {
    arrow_button* ab = scrl[i];
    ab->set_pos (x, y);
    x = x + w + dw[i];
  }

  x += w;
  cb_show_pitch_volume_board.set_pos (x, ab_scroll_down.extents.bottom - fnt.lift);
  cb_show_pitch_volume_board.set_text ("i");

  x += w;
  cb_show_pitch_volume_drones.set_pos (x + 1, ab_scroll_down.extents.bottom - fnt.lift);
  cb_show_pitch_volume_drones.set_text ("j");

  d_parameters.set_text ("Parameters");

  sp_attack_time.update ();
  sp_decay_time.update ();
  sp_voices.update ();
  sp_pitch_bend.update ();
  cb_show_nearby_notes.set_text ("Show nearby notes");

  l_waveform_display.update ();

  l_octave_shift.update ();
  ab_octave_up.update ();
  ab_octave_down.update ();
  sp_octave_shift_bpm.update ();

  settings_scr.update_widgets ();

}

void ui_list::dofft () {
  if (crved && crved->num_curves && !fft0.folded ()) fft0.go (crved->curveinfo[0].curve);
}

void fade_button_listener::after_fade (fader& f) {
  cb->enabled = 1;
  f.afl = 0;
}

void voice_listener::changed (checkbutton& cb) {
  fade_button_listener::changed (cb);
  MENU.next_tab = MENUP.cb_mkb_voice;
  MENU.next_tab_instr = &din0;
}

fade_button_listener::fade_button_listener (checkbutton* _cb, fader* _f, int* _t) {cb = _cb; f = _f; target = _t; lsnr = 0;}

void fade_button_listener::changed (checkbutton& cb) {
  if (f->on == 0) {
    int on = cb.is_on ();
    *target = on;
    if (on) f->set (0, 1); else f->set (1, 0);
    cb.enabled = 0;
    f->afl = this;
  }
}

void compress__listener::changed (checkbutton& cb) {
  din0.dinfo.compress = cb.is_on ();
}

void settings__listener::clicked (button& b) {
  uis.set_current (&uis.settings_scr);
}

void pitch_bend_listener::changed (field& f) {
  PITCH_BEND_PER_PIXEL = f;
  PITCH_BEND = PITCH_BEND_PER_PIXEL * 100;
}

void pitch_bend_listener::changed (checkbutton& cb) {
  keybd2.show_nearby_notes = cb.state;
}

void waveform_display_listener::clicked (button& b) {
  CRVED = &keybd2.waved;
  if (&b == &uis.ab_prev_wav) {
    CRVED->win.calc ();
    CRVED->load_curve (-1);
  } else {
    CRVED->win.calc ();
    CRVED->load_curve (+1);
  }
}

void parameters_listener::clicked (button& b) {
  arrow_button& ab = dynamic_cast<arrow_button&> (b);
  if (ab.dir == arrow_button::down) {
    ab.set_dir (arrow_button::right);
    uis.d_parameters.hide (widget::only_children);
  } else {
    uis.d_parameters.show ();
    ab.set_dir (arrow_button::down);
  }
  ab.show ();
}

ui_list::~ui_list () {

  dlog << "--- destroying uis ---" << endl;

  DEFINE_PARAMETERS
  widget_save ("d_parameters", pw, npars);

  widget* dpeuw [] = {&dpeu.depth, &dpeu.bpm};
  widget_save ("d_dpeu", dpeuw, 2);

  dlog << "--- destroyed uis ---" << endl;

}

void ui::enter () {
  warp_mouse (prev_mousex, prev_mousey);
}

void ui::leave () {
  if (hide_menu () == 0) {
    prev_mousex = mousex;
    prev_mousey = mousey;
  }
  if (!ed) scope.save_current_instrument ();
  if (mouse_slider0.active) mouse_slider0.deactivate ();
}

extern const float MIN_TIME;

void attack_val::changed (field& F) {
  float f = F;
  if (equals (f, 0.0f)) ATTACK_TIME = MIN_TIME; else ATTACK_TIME = f;
}

void decay_val::changed (field& F) {
  float f = F;
  if (equals (f, 0.0f)) DECAY_TIME = MIN_TIME; else DECAY_TIME = f;
}

void voices_val::changed (field& F) {
  int i = F;
  int j = i; if (i < 1) j = 1;
  NOTE_VOLUME = 1.0f / j;
  keybd2.calc_visual_params ();
}

void scroll_arrow_listener::clicked (button& b) {
  if (&b == &uis.ab_scroll_left) {
    din0.scroll (-din0.dinfo.scroll.dx, 0, 0);
  } else if (&b == &uis.ab_scroll_right) {
    din0.scroll (din0.dinfo.scroll.dx, 0, 0);
  } else if (&b == &uis.ab_scroll_up) {
    din0.scroll (0, -din0.dinfo.scroll.dy, 0);
  } else {
    din0.scroll (0, din0.dinfo.scroll.dy, 0);
  }
}

void show_pitch_volume_listener::changed (checkbutton& cb) {
  if (&cb == &uis.cb_show_pitch_volume_board)
    din0.dinfo.show_pitch_volume.board = cb.state;
  else
    din0.dinfo.show_pitch_volume.drones = cb.state;
}

void fft::clicked (button& b) {
  if (ab_fold.dir == arrow_button::down) {
    ab_fold.set_dir (arrow_button::right);
    l_title.hide (widget::only_children);
    ab_fold.show ();
  } else {
    ab_fold.set_dir (arrow_button::down);
    l_title.show ();
    uis.dofft ();
  }
}

int fft::folded () {
  return (ab_fold.dir == arrow_button::right);
}

void ui_list::handle_plugin (widget* which, int what) {
  enum {INSTALL = 1, REMOVE = 0};
  ui* ueds [] = {
    &din0.waved,
    &din0.moded,
    &din0.droneed,
    &keybd2.waved,
    &fractaliser_.ed
  };

  int num_editors = 5;
  if (what == INSTALL) {
    for (int i = 0; i < num_editors; ++i) widgets_of[ueds[i]].push_back (which);
  } else {
    for (int i = 0; i < num_editors; ++i) {
      vector<widget*>& widgets = widgets_of [ueds[i]];
      vector<widget*>::iterator iter = find (widgets.begin (), widgets.end(), which);
      if (iter != widgets.end()) widgets.erase (iter);
    }
  }
  main_menu.toggle ();
}

int ui_list::update_bottom_line () {
  widget** winst = 0;
  widget* wdin[] = {&cb_voice, &cb_gater, &cb_delay, &cb_compress, &main_menu.b_menu, &b_settings, &cb_record};
  widget* wmon[] = {&l_mondrian_voices, &cb_delay, &cb_compress,  &main_menu.b_menu, &b_settings, &cb_record};
  widget* wcom[] = {&cb_delay, &cb_compress, &main_menu.b_menu, &b_settings, &cb_record};
  widget* wed [] = {&main_menu.b_menu, &b_settings};
  instrument* inst = get_current_instrument ();
  int n = 0;
  if (inst == &din0) {
    winst = wdin;
    n = 7;
  } else if (inst == &mondrian0) {
    winst = wmon;
    n = 6;
  } else if (inst == &binaural_drones0 || inst == &keybd2) {
    winst = wcom;
    n = 5;
  } else {
    winst = wed;
    n = 2;
  }
  int dx = 25, w = 0;
  for (int i = 0; i < n; ++i) w += winst[i]->extents.width;
  w += n * dx;
  int x = (view.xmax - w) / 2, y = 2; //(int)(0.25f * get_line_height() + 0.5);
  for (int i = 0; i < n; ++i) {
    widget& b = *winst[i];
    const box<int>& e = b.extents;
    b.set_extents (x, y, x + e.width, y + e.height);
    b.set_pos (x, y);
    x = e.right + dx;
  }

  ol_voice_is.set_pos (cb_voice.posx - 3 * ol_voice_is.left.extents.width - 2, cb_voice.posy);
  return y;
}

int ui_list::remove (widget* w) {
  return erase (widgets_of [current], w);
}

void ui_list::add (ui* u, widget* w) { widgets_of [u].push_back (w); }

int ui_list::escape_from_things () {
  int ret = 1;
  if (UI_OFF) turn_on_ui ();
  else if (mkb_selector()) mkb_selector.abort ();
  else if (mon_selector()) mon_selector.abort ();
  else if (mouse_slider0.active) mouse_slider0.deactivate ();
  else if (esc_octave_shift(get_current_instrument()));
  else if (main_menu.show) main_menu.toggle ();
  else if (main_menu.b_close.visible) main_menu.b_close.call_listener ();
  else if (current->ed) ret = current->esc();
  else if (current == &din0) {
    if (din0.moving_drones) din0.toggle_moving_drones ();
    else if (din0.adding_drones) din0.toggle_adding_drones ();
    else if (din0.meshh.create) din0.stop_creating_mesh ();
    else if (din0.create_drone_pend) din0.stop_creating_drone_pendulum ();  
    else if (din0.finish_phrase_recording());
    else if (din0.dinfo.gravity.stop_editing ());
    else ret = 0;
  }
  else if (keybd2.turn_off_bend ());
  else if (mondrian0.stop_doing_stuff ());
  else if (binaural_drones0.aborted ());
  else if (current == &settings_scr) settings_scr.clicked (settings_scr.b_go_back);
  else ret = 0;
  return ret;
}

int is_menu_visible () {
  return uis.main_menu.show;
}

void abort_selectors () {
  mon_selector.abort ();
  mkb_selector.abort ();
}

PICKED_OPTION (ui_list,trig_what_lis) {
  int& tw = keybd2.trig_what;
  tw = !tw;
  extern const char* keys_trig_what [];
  set_label (l, keys_trig_what, tw);
}

PICKED_OPTION (ui_list, ol_voice_is_lis) {
  din0.dinfo.voice_is_voice = !din0.dinfo.voice_is_voice;
  extern ui_list uis;
  const char* viv = voice_is_lbls [din0.dinfo.voice_is_voice];
  uis.cb_voice.set_text (viv);
  MENU.handle_voice_tab_items (viv);
}

VALUE_CHANGED (ui_list, dpeu_lis) {
  din0.update_drone_pendulums ();
}