2003-10-16 14:02:09 +00:00
|
|
|
/***************************************************************************
|
|
|
|
graph.cpp - description
|
|
|
|
-------------------
|
|
|
|
begin : Thu Oct 2 2003
|
|
|
|
copyright : (C) 2003 by Michael Margraf
|
2004-06-16 17:41:33 +00:00
|
|
|
email : michael.margraf@alumni.tu-berlin.de
|
2003-10-16 14:02:09 +00:00
|
|
|
***************************************************************************/
|
|
|
|
|
|
|
|
/***************************************************************************
|
|
|
|
* *
|
|
|
|
* 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. *
|
|
|
|
* *
|
|
|
|
***************************************************************************/
|
|
|
|
|
|
|
|
#include "graph.h"
|
|
|
|
|
2004-04-04 15:56:10 +00:00
|
|
|
#include <math.h>
|
2004-08-25 19:46:13 +00:00
|
|
|
#include <stdlib.h>
|
2004-04-04 15:56:10 +00:00
|
|
|
|
|
|
|
|
2003-10-20 08:30:34 +00:00
|
|
|
Graph::Graph(const QString& _Line)
|
2003-10-16 14:02:09 +00:00
|
|
|
{
|
2004-04-04 15:56:10 +00:00
|
|
|
Type = isGraph;
|
|
|
|
|
2004-05-09 16:43:54 +00:00
|
|
|
Var = _Line;
|
2004-05-16 15:02:50 +00:00
|
|
|
countY = 0; // no points in graph
|
2004-08-29 14:51:07 +00:00
|
|
|
Points = 0;
|
2004-06-21 08:22:13 +00:00
|
|
|
Thick = numMode = 0;
|
2004-08-29 14:51:07 +00:00
|
|
|
Color = 0x0000ff; // blue
|
2004-06-21 08:22:13 +00:00
|
|
|
Style = 0; // solid line
|
|
|
|
Precision = 3;
|
2004-04-04 15:56:10 +00:00
|
|
|
isSelected = false;
|
|
|
|
|
2004-08-29 14:51:07 +00:00
|
|
|
Points = 0;
|
2004-05-09 16:43:54 +00:00
|
|
|
cPointsY = 0;
|
|
|
|
|
2004-08-20 20:13:01 +00:00
|
|
|
Markers.setAutoDelete(true);
|
2004-05-09 16:43:54 +00:00
|
|
|
cPointsX.setAutoDelete(true);
|
2003-10-16 14:02:09 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
Graph::~Graph()
|
|
|
|
{
|
2004-08-25 19:46:13 +00:00
|
|
|
if(Points != 0) free(Points);
|
2004-07-18 18:48:51 +00:00
|
|
|
if(cPointsY != 0) delete[] cPointsY;
|
2003-10-16 14:02:09 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// ---------------------------------------------------------------------
|
2004-10-10 16:06:55 +00:00
|
|
|
void Graph::paint(ViewPainter *p, int x0, int y0)
|
2003-10-16 14:02:09 +00:00
|
|
|
{
|
2004-08-29 14:51:07 +00:00
|
|
|
int *pp = Points;
|
|
|
|
if(pp == 0) {
|
2004-10-10 16:06:55 +00:00
|
|
|
p->Painter->setPen(QPen(QColor(Color))); // set color for xlabel text
|
2004-04-17 15:41:20 +00:00
|
|
|
return;
|
|
|
|
}
|
2004-04-04 15:56:10 +00:00
|
|
|
|
2004-08-29 14:51:07 +00:00
|
|
|
int n1;
|
2004-04-04 15:56:10 +00:00
|
|
|
if(isSelected) {
|
2004-10-10 16:06:55 +00:00
|
|
|
p->Painter->setPen(QPen(QPen::darkGray,Thick+4));
|
2004-05-16 15:02:50 +00:00
|
|
|
for(n1=countY; n1>0; n1--) {
|
2004-08-29 14:51:07 +00:00
|
|
|
p->drawPoint(x0+(*pp), y0-(*(pp+1)));
|
2004-06-21 08:22:13 +00:00
|
|
|
do {
|
2004-08-29 14:51:07 +00:00
|
|
|
pp += 2;
|
|
|
|
if(*pp > -2)
|
2004-06-21 08:22:13 +00:00
|
|
|
do {
|
2004-08-29 14:51:07 +00:00
|
|
|
p->drawLine(x0+(*(pp-2)), y0-(*(pp-1)), x0+(*pp), y0-(*(pp+1)));
|
|
|
|
pp += 2;
|
|
|
|
} while(*pp > -2); // until end of stroke
|
|
|
|
if(*pp < -9) break; // end of line ?
|
|
|
|
pp++;
|
|
|
|
} while(*pp > -9);
|
|
|
|
pp++;
|
2004-04-04 15:56:10 +00:00
|
|
|
}
|
|
|
|
|
2004-08-29 14:51:07 +00:00
|
|
|
pp = Points;
|
2004-10-10 16:06:55 +00:00
|
|
|
p->Painter->setPen(QPen(QPen::white, Thick, Qt::SolidLine));
|
2004-05-16 15:02:50 +00:00
|
|
|
for(n1=countY; n1>0; n1--) {
|
2004-08-29 14:51:07 +00:00
|
|
|
p->drawPoint(x0+(*pp), y0-(*(pp+1)));
|
2004-06-21 08:22:13 +00:00
|
|
|
do {
|
2004-08-29 14:51:07 +00:00
|
|
|
pp += 2;
|
|
|
|
if(*pp > -2)
|
2004-06-21 08:22:13 +00:00
|
|
|
do {
|
2004-08-29 14:51:07 +00:00
|
|
|
p->drawLine(x0+(*(pp-2)), y0-(*(pp-1)), x0+(*pp), y0-(*(pp+1)));
|
|
|
|
pp += 2;
|
|
|
|
} while(*pp > -2); // until end of stroke
|
|
|
|
if(*pp < -9) break; // end of line ?
|
|
|
|
pp++;
|
|
|
|
} while(*pp > -9);
|
|
|
|
pp++;
|
2004-04-04 15:56:10 +00:00
|
|
|
}
|
2004-10-10 16:06:55 +00:00
|
|
|
p->Painter->setPen(QPen(QColor(Color))); // set color for xlabel text
|
2004-04-04 15:56:10 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2004-05-09 16:43:54 +00:00
|
|
|
// **** not selected ****
|
2004-10-10 16:06:55 +00:00
|
|
|
p->Painter->setPen(QPen(QColor(Color), Thick, Qt::SolidLine));
|
2004-05-16 15:02:50 +00:00
|
|
|
for(n1=countY; n1>0; n1--) {
|
2004-08-29 14:51:07 +00:00
|
|
|
p->drawPoint(x0+(*pp), y0-(*(pp+1)));
|
2004-06-21 08:22:13 +00:00
|
|
|
do {
|
2004-08-29 14:51:07 +00:00
|
|
|
pp += 2;
|
|
|
|
if(*pp > -2)
|
2004-06-21 08:22:13 +00:00
|
|
|
do {
|
2004-08-29 14:51:07 +00:00
|
|
|
p->drawLine(x0+(*(pp-2)), y0-(*(pp-1)), x0+(*pp), y0-(*(pp+1)));
|
|
|
|
pp += 2;
|
|
|
|
} while(*pp > -2); // until end of stroke
|
|
|
|
if(*pp < -9) break; // end of line ?
|
|
|
|
pp++;
|
|
|
|
} while(*pp > -9);
|
|
|
|
pp++;
|
2003-10-16 14:02:09 +00:00
|
|
|
}
|
|
|
|
}
|
2003-10-20 08:30:34 +00:00
|
|
|
|
|
|
|
// ---------------------------------------------------------------------
|
|
|
|
QString Graph::save()
|
|
|
|
{
|
2004-06-22 16:49:55 +00:00
|
|
|
QString s = "\t<\""+Var+"\" "+Color.name()+
|
2004-06-16 17:41:33 +00:00
|
|
|
" "+QString::number(Thick)+" "+QString::number(Precision)+
|
2004-06-21 08:22:13 +00:00
|
|
|
" "+QString::number(numMode)+" "+QString::number(Style)+">";
|
2004-08-20 20:13:01 +00:00
|
|
|
|
|
|
|
for(Marker *pm=Markers.first(); pm != 0; pm=Markers.next())
|
|
|
|
s += "\n\t "+pm->save();
|
|
|
|
|
2003-10-20 08:30:34 +00:00
|
|
|
return s;
|
|
|
|
}
|
|
|
|
|
|
|
|
// ---------------------------------------------------------------------
|
|
|
|
bool Graph::load(const QString& _s)
|
|
|
|
{
|
|
|
|
bool ok;
|
|
|
|
QString s = _s;
|
|
|
|
|
|
|
|
if(s.at(0) != '<') return false;
|
|
|
|
if(s.at(s.length()-1) != '>') return false;
|
|
|
|
s = s.mid(1, s.length()-2); // cut off start and end character
|
|
|
|
|
2004-09-11 16:55:12 +00:00
|
|
|
Var = s.section('"',1,1); // Var
|
2003-10-20 08:30:34 +00:00
|
|
|
|
|
|
|
QString n;
|
|
|
|
n = s.section(' ',1,1); // Color
|
|
|
|
Color.setNamedColor(n);
|
|
|
|
if(!Color.isValid()) return false;
|
|
|
|
|
|
|
|
n = s.section(' ',2,2); // Thick
|
|
|
|
Thick = n.toInt(&ok);
|
|
|
|
if(!ok) return false;
|
|
|
|
|
2004-06-16 17:41:33 +00:00
|
|
|
n = s.section(' ',3,3); // Precision
|
|
|
|
Precision = n.toInt(&ok);
|
|
|
|
if(!ok) return false;
|
|
|
|
|
|
|
|
n = s.section(' ',4,4); // numMode
|
|
|
|
numMode = n.toInt(&ok);
|
|
|
|
if(!ok) return false;
|
|
|
|
|
2004-06-21 08:22:13 +00:00
|
|
|
n = s.section(' ',5,5); // Style
|
|
|
|
Style = n.toInt(&ok);
|
|
|
|
if(!ok) return false;
|
|
|
|
|
2003-10-20 08:30:34 +00:00
|
|
|
return true;
|
|
|
|
}
|
2004-04-04 15:56:10 +00:00
|
|
|
|
2004-05-07 17:31:21 +00:00
|
|
|
// -----------------------------------------------------------------------
|
|
|
|
// Checks if the coordinates x/y point to the graph. x/y are relative to
|
|
|
|
// diagram cx/cy. 5 is the precision the user must point onto the graph.
|
2004-04-04 15:56:10 +00:00
|
|
|
int Graph::getSelected(int x, int y)
|
|
|
|
{
|
2004-08-29 14:51:07 +00:00
|
|
|
int *pp = Points;
|
|
|
|
if(pp == 0) return -1;
|
2004-04-04 15:56:10 +00:00
|
|
|
|
2004-08-29 14:51:07 +00:00
|
|
|
int A, zi;
|
2004-04-10 11:29:03 +00:00
|
|
|
int dx, dx2, x1;
|
|
|
|
int dy, dy2, y1;
|
|
|
|
|
2004-05-16 15:02:50 +00:00
|
|
|
int countX = cPointsX.getFirst()->count;
|
|
|
|
for(int z=0; z<countY; z++) {
|
2004-06-21 08:22:13 +00:00
|
|
|
zi = 0;
|
|
|
|
do {
|
2004-08-29 14:51:07 +00:00
|
|
|
x1 = *(pp++); y1 = *(pp++);
|
2004-06-21 08:22:13 +00:00
|
|
|
zi++;
|
2004-04-10 11:29:03 +00:00
|
|
|
|
2004-05-16 15:02:50 +00:00
|
|
|
dx = x - x1;
|
2004-08-29 14:51:07 +00:00
|
|
|
dx2 = (*pp);
|
2004-06-21 08:22:13 +00:00
|
|
|
if(dx2 < -1) {
|
|
|
|
if(dx2 < -9) break;
|
2004-08-29 14:51:07 +00:00
|
|
|
dx2 = *(++pp);
|
2004-06-21 08:22:13 +00:00
|
|
|
if(dx2 < -9) break;
|
|
|
|
zi -= 2; // because of space (edge-values)
|
|
|
|
}
|
2004-05-16 15:02:50 +00:00
|
|
|
if(dx < -5) { if(x < dx2-5) continue; } // point between x coordinates ?
|
2004-06-16 17:41:33 +00:00
|
|
|
else { if(x > 5) if(x > dx2+5) continue; }
|
2004-04-10 11:29:03 +00:00
|
|
|
|
2004-05-16 15:02:50 +00:00
|
|
|
dy = y - y1;
|
2004-08-29 14:51:07 +00:00
|
|
|
dy2 = (*(pp+1));
|
2004-05-16 15:02:50 +00:00
|
|
|
if(dy < -5) { if(y < dy2-5) continue; } // point between y coordinates ?
|
2004-06-16 17:41:33 +00:00
|
|
|
else { if(y > 5) if(y > dy2+5) continue; }
|
2004-04-10 11:29:03 +00:00
|
|
|
|
2004-05-16 15:02:50 +00:00
|
|
|
dx2 -= x1;
|
|
|
|
dy2 -= y1;
|
2004-04-10 11:29:03 +00:00
|
|
|
|
2004-05-16 15:02:50 +00:00
|
|
|
A = dx2*dy - dx*dy2; // calculate the rectangle area spanned
|
|
|
|
A *= A; // avoid the need for square root
|
|
|
|
A -= 25*(dx2*dx2 + dy2*dy2); // substract selectable area
|
2004-04-10 11:29:03 +00:00
|
|
|
|
2004-06-21 08:22:13 +00:00
|
|
|
if(A <= 0) return z*countX + zi; // lies x/y onto the graph line ?
|
|
|
|
} while(true);
|
2004-08-29 14:51:07 +00:00
|
|
|
pp++;
|
2004-04-10 11:29:03 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return -1;
|
2004-04-04 15:56:10 +00:00
|
|
|
}
|
2004-08-14 06:40:55 +00:00
|
|
|
|
|
|
|
// -----------------------------------------------------------------------
|
|
|
|
// Creates a new graph and copies all the properties into it.
|
|
|
|
Graph* Graph::sameNewOne()
|
|
|
|
{
|
|
|
|
Graph *pg = new Graph(Var);
|
|
|
|
|
|
|
|
pg->Color = Color;
|
|
|
|
pg->Thick = Thick;
|
|
|
|
pg->Style = Style;
|
|
|
|
|
|
|
|
pg->Precision = Precision;
|
|
|
|
pg->numMode = numMode;
|
|
|
|
|
2004-08-20 20:13:01 +00:00
|
|
|
for(Marker *pm = Markers.first(); pm != 0; pm = Markers.next())
|
|
|
|
pg->Markers.append(pm->sameNewOne(pg));
|
|
|
|
|
2004-08-14 06:40:55 +00:00
|
|
|
return pg;
|
|
|
|
}
|