qucs_s/qucs/diagrams/graph.cpp

199 lines
5.5 KiB
C++
Raw Normal View History

2003-10-16 14:02:09 +00:00
/***************************************************************************
graph.cpp - description
-------------------
begin : Thu Oct 2 2003
copyright : (C) 2003 by Michael Margraf
email : margraf@mwt.ee.tu-berlin.de
***************************************************************************/
/***************************************************************************
* *
* 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>
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
2003-10-20 08:30:34 +00:00
Points = 0;
2004-05-09 16:43:54 +00:00
Thick = 0;
2003-11-03 10:27:28 +00:00
Color = 0x0000ff; // blue
2004-04-04 15:56:10 +00:00
isSelected = false;
2003-10-28 11:44:24 +00:00
Points = 0;
2004-05-09 16:43:54 +00:00
cPointsY = 0;
cPointsX.setAutoDelete(true);
2003-10-16 14:02:09 +00:00
}
Graph::~Graph()
{
2003-10-28 11:44:24 +00:00
if(Points != 0) delete[] Points;
2003-10-16 14:02:09 +00:00
}
// ---------------------------------------------------------------------
2004-04-17 09:11:48 +00:00
void Graph::paint(QPainter *p, int x0, int y0)
2003-10-16 14:02:09 +00:00
{
int *pp = Points;
2004-04-17 15:41:20 +00:00
if(pp == 0) {
p->setPen(QPen(QColor(Color))); // set color for xlabel text
return;
}
2004-04-04 15:56:10 +00:00
2004-05-16 15:02:50 +00:00
int countX = cPointsX.getFirst()->count;
2004-05-09 16:43:54 +00:00
int n1, n2;
2004-04-04 15:56:10 +00:00
if(isSelected) {
p->setPen(QPen(QPen::darkGray,Thick+4));
2004-05-16 15:02:50 +00:00
for(n1=countY; n1>0; n1--) {
2004-05-09 16:43:54 +00:00
p->drawPoint(x0+(*pp), y0-(*(pp+1)));
2004-05-16 15:02:50 +00:00
for(n2=countX-1; n2>0; n2--) {
2004-05-09 16:43:54 +00:00
p->drawLine(x0+(*pp), y0-(*(pp+1)), x0+(*(pp+2)), y0-(*(pp+3)));
pp += 2;
}
2004-04-04 15:56:10 +00:00
pp += 2;
}
pp = Points;
p->setPen(QPen(QPen::white, Thick, Qt::SolidLine));
2004-05-16 15:02:50 +00:00
for(n1=countY; n1>0; n1--) {
2004-05-09 16:43:54 +00:00
p->drawPoint(x0+(*pp), y0-(*(pp+1)));
2004-05-16 15:02:50 +00:00
for(n2=countX-1; n2>0; n2--) {
2004-05-09 16:43:54 +00:00
p->drawLine(x0+(*pp), y0-(*(pp+1)), x0+(*(pp+2)), y0-(*(pp+3)));
pp += 2;
}
2004-04-04 15:56:10 +00:00
pp += 2;
}
2004-04-17 15:41:20 +00:00
p->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 ****
2003-11-03 10:27:28 +00:00
p->setPen(QPen(QColor(Color), Thick, Qt::SolidLine));
2004-05-16 15:02:50 +00:00
for(n1=countY; n1>0; n1--) {
2004-05-09 16:43:54 +00:00
p->drawPoint(x0+(*pp), y0-(*(pp+1)));
2004-05-16 15:02:50 +00:00
for(n2=countX-1; n2>0; n2--) {
2004-05-09 16:43:54 +00:00
p->drawLine(x0+(*pp), y0-(*(pp+1)), x0+(*(pp+2)), y0-(*(pp+3)));
pp += 2;
}
2003-10-16 14:02:09 +00:00
pp += 2;
}
}
2003-10-20 08:30:34 +00:00
// ---------------------------------------------------------------------
QString Graph::save()
{
2004-05-16 15:02:50 +00:00
QString s = " <\""+Var+"\" "+Color.name()+
" "+QString::number(Thick)+">";
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-05-16 15:02:50 +00:00
if(s.at(0) == '"') Var = s.section('"',1,1); // Var, the new style
else Var = s.section(' ',0,0); // Var, the old style
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;
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)
{
int *pp = Points;
if(pp == 0) return -1;
2004-04-10 11:29:03 +00:00
int A;
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++) {
for(int n=1; n<countX; n++) { // count-1 runs (need 2 points per run)
x1 = *(pp++); y1 = *(pp++);
2004-04-10 11:29:03 +00:00
2004-05-16 15:02:50 +00:00
dx = x - x1;
dx2 = (*pp);
if(dx < -5) { if(x < dx2-5) continue; } // point between x coordinates ?
else { if(x > dx2+5) continue; }
2004-04-10 11:29:03 +00:00
2004-05-16 15:02:50 +00:00
dy = y - y1;
dy2 = (*(pp+1));
if(dy < -5) { if(y < dy2-5) continue; } // point between y coordinates ?
else { 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-05-16 15:02:50 +00:00
if(A <= 0) return z*countX + n; // lies x/y onto the graph line ?
}
pp += 2;
2004-04-10 11:29:03 +00:00
}
return -1;
/* ------------------------------------------------------------
This algorithm is not bad, but not very fast.
------------------------------------------------------------
2004-04-04 15:56:10 +00:00
double x_double, y_double, phi, sin_phi, cos_phi;
int len, xn, yn;
int x2, x1 = *(pp++);
int y2, y1 = *(pp++);
2004-05-07 17:31:21 +00:00
for(int n=1; n<count; n++) { // count-1 runs (need 2 points per run)
2004-04-04 15:56:10 +00:00
x2 = (*pp) - x1;
y2 = (*(pp+1)) - y1;
x_double = double(x-x1);
y_double = double(y-y1);
phi = -atan2(double(y2), double(x2));
sin_phi = sin(phi);
cos_phi = cos(phi);
len = int( sqrt(double( x2*x2 + y2*y2 )) )+5;
xn = int(x_double*cos_phi - y_double*sin_phi);
yn = int(x_double*sin_phi + y_double*cos_phi);
// lies x/y onto the graph line ?
if(xn > -5) if(xn < len) if(yn > -5) if(yn < 5) return n;
x1 = *(pp++); y1 = *(pp++);
}
return -1;
2004-04-10 11:29:03 +00:00
------------------------------------------------------------ */
2004-04-04 15:56:10 +00:00
}