Blob Blame Raw
/*
 *   HDSPMixer
 *    
 *   Copyright (C) 2003 Thomas Charbonnel (thomas@undata.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., 675 Mass Ave, Cambridge, MA 02139, USA.
 */

#pragma implementation
#include "HDSPMixerFader.h"

HDSPMixerFader::HDSPMixerFader(int x, int y, double r, int id, int src):Fl_Widget(x, y, 13, 153)
{
    index = id;
    source = src;
    ndb = (int)(CF*(double)(log(0.5*(exp(3.0)-1)+1)*(double)(FADER_HEIGHT)/3.0));
    non_submix_dest = 0;
    dest = 0;
    ref = r;
    basew = (HDSPMixerWindow *)window();
    anchor = lastpos = lasty = drag = shift_orig = y_orig = 0;
    for (int i = 0; i < HDSP_MAX_DEST; i++) {
	pos[i] = 0;
    }
}

void HDSPMixerFader::draw()
{
    fl_draw_pixmap(Slider1_xpm, x(), y()+139-(int)(pos[dest]/CF));
}

int HDSPMixerFader::handle(int e)
{
    int button3 = Fl::event_button3();
    int shift = Fl::event_shift();
    int ctrl = Fl::event_ctrl();
    int ypos = Fl::event_y()-y();
    switch (e) {
	case FL_PUSH:
	    if (onSlider(ypos)) {
		anchor = 144-ypos-(int)(pos[dest]/CF);
		if (button3) relative->set(pos[dest]);
		return 1;
	    }
	    anchor = 0;
	    if (ctrl) {
		pos[dest] = ndb;
	    } else if (ypos < 7) {
		pos[dest] = 137*CF;
	    } else if (ypos > 6 && ypos < 146) {
		pos[dest] = (144-ypos)*CF;
	    } else if (ypos > 145) {
		pos[dest] = 0;
	    } 
	    if (lastpos != pos[dest]) {
		basew->setMixer(index, source, dest);
		sendGain();
		lastpos = pos[dest];
	    }
	    if (lasty != (int)(pos[dest]/CF)) {
		redraw();
		lasty = (int)(pos[dest]/CF);
	    }
	    if (button3) relative->set(pos[dest]);
	    shift_orig = pos[dest];
	    y_orig = ypos;
	    basew->checkState();
	    return 1;
	case FL_DRAG:
	    ypos += anchor;
	    if (ctrl) {
		pos[dest] = ndb;
		shift_orig = pos[dest];
		y_orig = ypos;
	    } else if ((ypos > 6 && ypos < 146) || drag) {
		drag = 1;
		if (shift) {
		    pos[dest] = (y_orig-ypos)+shift_orig;
		    if (pos[dest] < 0) pos[dest] = 0;
		    if (pos[dest] > 137*CF) pos[dest] = 137*CF;
		} else {
		    if (ypos < 7) {
			pos[dest] = 137*CF;
		    } else if (ypos > 144) {
			pos[dest] = 0;
		    } else {
			pos[dest] = (144-ypos)*CF;
		    }
		    shift_orig = pos[dest];   
		    y_orig = ypos;
		}
	    }
	    if (lastpos != pos[dest]) {
		basew->setMixer(index, source, dest);
		sendGain();
		lastpos = pos[dest];
	    }
	    if (lasty != (int)(pos[dest]/CF)) {
		redraw();
		lasty = (int)(pos[dest]/CF);
	    }
	    if (button3) relative->set(pos[dest]);
	    basew->checkState();
	    return 1;
	case FL_RELEASE:
	    drag = 0;
	    anchor = 0;
	    return 1;   
	default :
	    return Fl_Widget::handle(e);
    }
}

int HDSPMixerFader::onSlider(int ypos)
{
    if (ypos > (139-(int)(pos[dest]/CF)) && ypos < (151-(int)(pos[dest]/CF))) {
	return 1;
    }
    return 0;
}

void HDSPMixerFader::set(int p)
{
    if (p != pos[dest]) {
	pos[dest] = lastpos = p;
	basew->setMixer(index, source, dest);
	sendGain();
    }
    if (lasty != (int)(pos[dest]/CF)) {
	lasty = (int)(pos[dest]/CF);
	redraw();
    }
}

int HDSPMixerFader::posToInt(int p) {
    double x, y;
    
    if (p == ndb) return 32768;
    if (p == 137*CF) return 65535;
    if (p == 0) return 0;
    
    x = ((double)(p)) / (double)(137*CF);
    y = 65535.0 * (exp(3.0 * x) - 1.0) / (exp(3.0) - 1.0);
    if (y > 65535.0) y = 65535.0;
    if (y < 0.0) y = 0.0;
    return (int)y;
}

void HDSPMixerFader::posToLog(char *s)
{
    double db, fpos;
    
    if (posToInt(pos[dest]) == 0) {
	snprintf(s, 10, "-oo");
	return;
    }    
    fpos = (double)posToInt(pos[dest]) / ref;
    db = 20.0 * log10(fpos);
    snprintf(s, 10, "%.1f", db);
}

void HDSPMixerFader::sendGain()
{
    char buf[10];
    posToLog(buf);
    gain->setText(buf);
}