Skip to content
Snippets Groups Projects
  • Stuart Layton's avatar
    167a58d7
    The spikeviewer shows up without segfaults! That was more painful than it should have been. · 167a58d7
    Stuart Layton authored
    The SpikeDisplayNode object isn't working properly. Queueing of spikes there appeared to be what was causing the problem.
    
    I need to consider an alternative way of handling the SpikeObjects. In the past I've used a circular buffer but that has
    problems with having a set size. I can use an stl::queue but queues are not thread safe, which is something that we
    definitely need.
    
    Anyway the plots show up and dance when the viewer is turned on.  I still need to implement a Stereotrode plot and then the
    tetrode plot. I'll probably derive them from the ElectrodePlot base class instead of creating a base classs from which all
    three plot types are derived from.
    
    Finally I need to get a better handle on the setup of the OpenGLCanvas and how its sized.  How the "Magical" scroll bars work
    and how to auto place the plots.
    
    Do we force all the plots to be of the same type? Or do we mix in Electrode Plots with Tetrode Plots?  If so how do we orient them all?
    167a58d7
    History
    The spikeviewer shows up without segfaults! That was more painful than it should have been.
    Stuart Layton authored
    The SpikeDisplayNode object isn't working properly. Queueing of spikes there appeared to be what was causing the problem.
    
    I need to consider an alternative way of handling the SpikeObjects. In the past I've used a circular buffer but that has
    problems with having a set size. I can use an stl::queue but queues are not thread safe, which is something that we
    definitely need.
    
    Anyway the plots show up and dance when the viewer is turned on.  I still need to implement a Stereotrode plot and then the
    tetrode plot. I'll probably derive them from the ElectrodePlot base class instead of creating a base classs from which all
    three plot types are derived from.
    
    Finally I need to get a better handle on the setup of the OpenGLCanvas and how its sized.  How the "Magical" scroll bars work
    and how to auto place the plots.
    
    Do we force all the plots to be of the same type? Or do we mix in Electrode Plots with Tetrode Plots?  If so how do we orient them all?
WaveAxes.cpp 4.89 KiB
#include "WaveAxes.h"

WaveAxes::WaveAxes():
					GenericAxes(),
					drawWaveformLine(true),
					drawWaveformPoints(false),
					drawGrid(true),
					overlay(false),
					convertLabelUnits(true)
{	
	GenericAxes::gotFirstSpike = false;
	GenericAxes::resizedFlag = false;

	ylims[0] = 0;
	ylims[1] = 1;
	setWaveformColor(1.0,1.0,0.6);
	setThresholdColor(1.0, 0.1, 0.1);
	setGridColor(0.4, 0.2, 0.2);
	
	BaseUIElement::elementName = (char*) "WaveAxes";
}

WaveAxes::WaveAxes(int x, int y, double w, double h, int t):
					GenericAxes(x,y,w,h,t),
					drawWaveformLine(true),
					drawWaveformPoints(false),
					drawGrid(true),
					overlay(false),
					convertLabelUnits(true)
{	
	GenericAxes::gotFirstSpike = false;
	GenericAxes::resizedFlag = false;

	setWaveformColor(1.0,1.0,0.6);
	setThresholdColor(1.0, 0.1, 0.1);
	setGridColor(0.2, 0.2, 0.2);
	BaseUIElement::elementName = (char*) "WaveAxes";

}

void WaveAxes::updateSpikeData(SpikeObject newSpike){
	std::cout<<"WaveAxes::updateSpikeData()"<<std::endl;
	GenericAxes::updateSpikeData(newSpike);

}

void WaveAxes::redraw(){
	
	BaseUIElement::redraw();
	
	plot();
	
	BaseUIElement::drawElementEdges();
}


void WaveAxes::plot(){

	int chan = 0;
	// If no spikes have been received then don't plot anything
	if (!gotFirstSpike)
	{
		std::cout<<"\tWaiting for the first spike"<<std::endl;
		return;
	}
	
	// Set the plotting range for the current axes the xlims member is ignored as the xdims are 0->number of samples per waveform minus one
    // so the line goes all the way to the edges ydims are specified by the ylims vector		
	setViewportRange(0, ylims[0], s.nSamples-1, ylims[1]);
	
	// draw the grid lines for the waveforms?
	if(drawGrid)
		drawWaveformGrid(s.threshold[chan], s.gain[chan]);
	
	//compute the spatial width for each wawveform sample	
	float dx = 1;
	float x = 0;
	int	sampIdx = 0; 

	//Draw the individual waveform points connected with a line
	// if drawWaveformPoints is set to false then force the drawing of the line, _SOMETHING_ must be drawn
	glColor3fv(waveColor);
	
	glLineWidth(1);
	glBegin( GL_LINE_STRIP );
	
	int dSamples = 1;
	for (int i=0; i<s.nSamples; i++)
	{
		//std::cout<<"\t"<<s.data[sampIdx];
		glVertex2f(x, s.data[sampIdx]);
		sampIdx += dSamples;
		x +=dx; 
	}
	
	glEnd();
	// std::cout<<std::endl;
	

	// Draw the threshold line and label
	
	glColor3fv(thresholdColor);
	glLineWidth(1); 
	glLineStipple(4, 0xAAAA); // make a dashed line
	glEnable(GL_LINE_STIPPLE);

	glBegin( GL_LINE_STRIP );
		glVertex2f(0, s.threshold[chan]);
		glVertex2f(s.nSamples, s.threshold[chan]);
	glEnd();		

	glDisable(GL_LINE_STIPPLE);

	char str[500] = {0};
	
	/*if(convertLabelUnits)
		sprintf(str, "%duV", ad16ToUv(s.threshold[chan], s.gain[chan]));
	else
		sprintf(str, "%d", (int) s.threshold[chan]);*/
	makeLabel(s.threshold[chan], s.gain[chan], convertLabelUnits, str);
	
//	printf(str);
	
	float yOffset = (ylims[1] - ylims[0])/BaseUIElement::height * 2;
	//drawString(1 ,s.threshold[chan] + yOffset, GLUT_BITMAP_8_BY_13, str);
}

void WaveAxes::drawWaveformGrid(int thold, int gain){

	double voltRange = ylims[1] - ylims[0];
	double pixelRange = BaseUIElement::height;
	//This is a totally arbitrary value that i'll mess around with and set as a macro when I figure out a value I like
	int minPixelsPerTick = 25;
	int MAX_N_TICKS = 10;

	int nTicks = pixelRange / minPixelsPerTick;
	while(nTicks>MAX_N_TICKS){
		minPixelsPerTick += 5;
		nTicks = pixelRange / minPixelsPerTick;
	}
	int voltPerTick = (voltRange / nTicks);
	// Round to the nearest 200

	
	double meanRange = voltRange/2;
	glColor3fv(gridColor);

	glLineWidth(1);
	char str[200] = {0}; 
	
	double tickVoltage = thold;
	while(tickVoltage < ylims[1] - voltPerTick/2) // Draw the ticks above the thold line
	{
		tickVoltage = roundUp(tickVoltage + voltPerTick, 100);
		
		glBegin(GL_LINE_STRIP);
		glVertex2i(0, tickVoltage);
		glVertex2i(s.nSamples, tickVoltage);
		glEnd();
		
		makeLabel(tickVoltage, gain, convertLabelUnits, str);
		//drawString(1, tickVoltage+voltPerTick/10, GLUT_BITMAP_8_BY_13, str);
	}
	
	tickVoltage = thold;
	while(tickVoltage > ylims[0] + voltPerTick) // draw the ticks below the thold line
	{
		tickVoltage = roundUp(tickVoltage - voltPerTick, 100);

		glBegin(GL_LINE_STRIP);
		glVertex2i(0, tickVoltage);
		glVertex2i(s.nSamples, tickVoltage);
		glEnd();
			
		makeLabel(tickVoltage, gain, convertLabelUnits, str);
		//drawString(1, tickVoltage+voltPerTick/10, GLUT_BITMAP_8_BY_13, str);
	}
	
	
}

void WaveAxes::setWaveformColor(GLfloat r, GLfloat g, GLfloat b){
	waveColor[0] = r;
	waveColor[1] = g;
	waveColor[2] = b;
}
void WaveAxes::setThresholdColor(GLfloat r, GLfloat g, GLfloat b){
	thresholdColor[0] = r;
	thresholdColor[1] = g;
	thresholdColor[2] = b;
}
// void WaveAxes::setPointColor(GLfloat r, GLfloat g, GLfloat b){
// 	pointColor[0] = r;
// 	pointColor[1] = g;
// 	pointColor[2] = b;
// }
void WaveAxes::setGridColor(GLfloat r, GLfloat g, GLfloat b){
	gridColor[0] = r;
	gridColor[1] = g;
	gridColor[2] = b;
}