qucs_s/qucs/diagrams/graph.cpp

247 lines
6.4 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
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-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-10-31 17:55:57 +00:00
yAxisNo = 0; // left y axis
2004-04-04 15:56:10 +00:00
2007-02-08 07:04:54 +00:00
ScrPoints = 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()
{
2007-02-08 07:04:54 +00:00
if(ScrPoints != 0)
free(ScrPoints);
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
{
2007-02-08 07:04:54 +00:00
if(ScrPoints == 0)
2004-04-17 15:41:20 +00:00
return;
2004-04-04 15:56:10 +00:00
if(isSelected) {
p->Painter->setPen(QPen(QPen::darkGray,Thick*p->PrintScale+4));
2004-12-04 18:41:22 +00:00
paintLines(p, x0, y0);
2004-04-04 15:56:10 +00:00
p->Painter->setPen(QPen(QPen::white, Thick*p->PrintScale, Qt::SolidLine));
2004-12-04 18:41:22 +00:00
paintLines(p, x0, y0);
2004-04-04 15:56:10 +00:00
return;
}
2004-05-09 16:43:54 +00:00
// **** not selected ****
p->Painter->setPen(QPen(QColor(Color), Thick*p->PrintScale, Qt::SolidLine));
2004-12-04 18:41:22 +00:00
paintLines(p, x0, y0);
}
// ---------------------------------------------------------------------
void Graph::paintLines(ViewPainter *p, int x0, int y0)
{
2007-02-08 07:04:54 +00:00
switch(Style) {
case GRAPHSTYLE_STAR:
p->drawStarSymbols(x0, y0, ScrPoints);
break;
case GRAPHSTYLE_CIRCLE:
p->drawCircleSymbols(x0, y0, ScrPoints);
break;
case GRAPHSTYLE_ARROW:
p->drawArrowSymbols(x0, y0, ScrPoints);
break;
default:
p->drawLines(x0, y0, ScrPoints);
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-10-31 17:55:57 +00:00
" "+QString::number(numMode)+" "+QString::number(Style)+
" "+QString::number(yAxisNo)+">";
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;
2004-10-31 17:55:57 +00:00
n = s.section(' ',6,6); // yAxisNo
if(n.isEmpty()) return true; // backward compatible
yAxisNo = 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)
{
2007-02-08 07:04:54 +00:00
float *pp = ScrPoints;
2004-08-29 14:51:07 +00:00
if(pp == 0) return -1;
2004-04-04 15:56:10 +00:00
2005-10-24 06:10:35 +00:00
int A, z=0;
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;
2007-02-08 07:04:54 +00:00
if(*pp <= STROKEEND) {
if(*pp <= BRANCHEND) z++;
2005-02-20 13:20:24 +00:00
pp++;
2007-02-08 07:04:54 +00:00
if(*pp <= BRANCHEND) {
if(*pp <= GRAPHEND) return -1; // not even one point ?
2005-05-09 06:32:17 +00:00
z++;
2005-02-20 13:20:24 +00:00
pp++;
2007-02-08 07:04:54 +00:00
if(*pp < BRANCHEND) return -1; // not even one point ?
2005-02-20 13:20:24 +00:00
}
}
2005-10-24 06:10:35 +00:00
2007-02-08 07:04:54 +00:00
if(Style >= GRAPHSTYLE_STAR) {
// for graph symbols
while(*pp > GRAPHEND) {
if(*pp > STROKEEND) {
dx = x - int(*(pp++));
dy = y - int(*(pp++));
if(dx < -5) continue;
if(dx > 5) continue;
if(dy < -5) continue;
if(dy > 5) continue;
return z*countX; // points on graph symbol
}
else {
z++; // next branch
pp++;
}
}
return -1;
}
2007-02-08 07:04:54 +00:00
// for graph lines
while(*pp > GRAPHEND) {
while(*pp >= STROKEEND) {
x1 = int(*(pp++));
y1 = int(*(pp++));
2004-05-16 15:02:50 +00:00
dx = x - x1;
2004-12-04 18:41:22 +00:00
dy = y - y1;
2007-02-08 07:04:54 +00:00
dx2 = int(*pp);
if(dx2 <= STROKEEND) { // end of stroke ?
if(dx2 <= BRANCHEND) break;
pp++;
dx2 = int(*pp); // go on as graph can also be selected between strokes
if(dx2 <= BRANCHEND) break;
2004-06-21 08:22:13 +00:00
}
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
2007-02-08 07:04:54 +00:00
dy2 = int(*(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
2005-10-24 06:10:35 +00:00
if(A <= 0) return z*countX; // lies x/y onto the graph line ?
}
2004-08-29 14:51:07 +00:00
pp++;
2005-05-02 06:28:45 +00:00
z++;
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-11-06 16:29:51 +00:00
pg->yAxisNo = yAxisNo;
2004-08-14 06:40:55 +00:00
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;
}