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-08-25 19:46:13 +00:00
|
|
|
#include <stdlib.h>
|
2015-07-16 11:32:05 +02:00
|
|
|
#include <iostream>
|
2004-04-04 15:56:10 +00:00
|
|
|
|
2014-11-04 12:59:41 +08:00
|
|
|
#include <QPainter>
|
2015-07-03 17:50:16 +02:00
|
|
|
#include <QDebug>
|
2004-04-04 15:56:10 +00:00
|
|
|
|
2015-06-29 22:02:30 +02:00
|
|
|
class Diagram;
|
|
|
|
|
|
|
|
Graph::Graph(Diagram const* d, const QString& _Line) :
|
|
|
|
Element(),
|
|
|
|
Style(GRAPHSTYLE_SOLID),
|
|
|
|
diagram(d)
|
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
|
|
|
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
|
|
|
|
2004-05-09 16:43:54 +00:00
|
|
|
cPointsY = 0;
|
2003-10-16 14:02:09 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
Graph::~Graph()
|
|
|
|
{
|
2007-02-08 07:04:54 +00:00
|
|
|
if(cPointsY != 0)
|
|
|
|
delete[] cPointsY;
|
2003-10-16 14:02:09 +00:00
|
|
|
}
|
|
|
|
|
2015-06-27 17:40:53 +02:00
|
|
|
// ---------------------------------------------------------------------
|
|
|
|
void Graph::createMarkerText() const
|
|
|
|
{
|
|
|
|
for(auto pm : Markers) {
|
|
|
|
pm->createText();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
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
|
|
|
{
|
2015-06-16 09:20:21 +02:00
|
|
|
if(!ScrPoints.size())
|
2004-04-17 15:41:20 +00:00
|
|
|
return;
|
2004-04-04 15:56:10 +00:00
|
|
|
|
|
|
|
if(isSelected) {
|
2012-10-31 09:15:06 +01:00
|
|
|
p->Painter->setPen(QPen(Qt::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
|
|
|
|
2012-10-31 09:15:06 +01:00
|
|
|
p->Painter->setPen(QPen(Qt::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 ****
|
2008-02-01 15:53:57 +00:00
|
|
|
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:
|
2015-06-16 09:20:21 +02:00
|
|
|
drawStarSymbols(x0, y0, p);
|
2007-02-08 07:04:54 +00:00
|
|
|
break;
|
|
|
|
case GRAPHSTYLE_CIRCLE:
|
2015-06-16 09:20:21 +02:00
|
|
|
drawCircleSymbols(x0, y0, p);
|
2007-02-08 07:04:54 +00:00
|
|
|
break;
|
|
|
|
case GRAPHSTYLE_ARROW:
|
2015-06-16 09:20:21 +02:00
|
|
|
drawArrowSymbols(x0, y0, p);
|
2007-02-08 07:04:54 +00:00
|
|
|
break;
|
|
|
|
default:
|
2015-06-16 09:20:21 +02:00
|
|
|
drawLines(x0, y0, p);
|
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
|
|
|
|
2014-09-03 15:52:40 +02:00
|
|
|
foreach(Marker *pm, Markers)
|
2004-08-20 20:13:01 +00:00
|
|
|
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
|
2015-06-18 10:59:59 +02:00
|
|
|
int st = n.toInt(&ok);
|
2004-06-21 08:22:13 +00:00
|
|
|
if(!ok) return false;
|
2015-06-18 10:59:59 +02:00
|
|
|
Style = toGraphStyle(st);
|
|
|
|
if(Style==GRAPHSTYLE_INVALID) return false;
|
2004-06-21 08:22:13 +00:00
|
|
|
|
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
|
|
|
// -----------------------------------------------------------------------
|
2015-06-27 17:40:53 +02:00
|
|
|
/*!
|
|
|
|
* Checks if the coordinates x/y point to the graph. returns the number of the
|
|
|
|
* branch of the graph, -1 upon a miss.
|
|
|
|
*
|
|
|
|
* x/y are relative to diagram cx/cy. 5 is the precision the user must point
|
|
|
|
* onto the graph.
|
|
|
|
*
|
|
|
|
* FIXME: should return reference to hit sample point or some context.
|
|
|
|
*/
|
2004-04-04 15:56:10 +00:00
|
|
|
int Graph::getSelected(int x, int y)
|
|
|
|
{
|
2015-06-16 09:20:21 +02:00
|
|
|
auto pp = ScrPoints.begin();
|
|
|
|
if(pp == ScrPoints.end()) 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;
|
|
|
|
|
2015-06-25 16:28:15 +02:00
|
|
|
int countX = cPointsX.at(0)->count;
|
2015-06-18 12:58:33 +02:00
|
|
|
if(pp->isStrokeEnd()) {
|
|
|
|
if(pp->isBranchEnd()) z++;
|
2005-02-20 13:20:24 +00:00
|
|
|
pp++;
|
2015-06-18 12:58:33 +02:00
|
|
|
if(pp->isBranchEnd()) {
|
|
|
|
if(pp->isGraphEnd()) return -1; // not even one point ?
|
2005-05-09 06:32:17 +00:00
|
|
|
z++;
|
2005-02-20 13:20:24 +00:00
|
|
|
pp++;
|
2015-06-18 12:58:33 +02:00
|
|
|
if(pp->isGraphEnd()) 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
|
2015-06-18 12:58:33 +02:00
|
|
|
while(!pp->isGraphEnd()) {
|
|
|
|
if(!pp->isStrokeEnd()) {
|
2015-06-30 10:25:32 +02:00
|
|
|
dx = x - int((pp)->getScrX());
|
|
|
|
dy = y - int((pp++)->getScrY());
|
2007-02-08 07:04:54 +00:00
|
|
|
|
|
|
|
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++;
|
|
|
|
}
|
|
|
|
}
|
2007-02-08 07:04:58 +00:00
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2007-02-08 07:04:54 +00:00
|
|
|
// for graph lines
|
2015-06-18 12:58:33 +02:00
|
|
|
while(!pp->isGraphEnd()) {
|
|
|
|
while(!pp->isBranchEnd()) {
|
2015-06-30 10:25:32 +02:00
|
|
|
x1 = int(pp->getScrX());
|
|
|
|
y1 = int((pp++)->getScrY());
|
2004-05-16 15:02:50 +00:00
|
|
|
dx = x - x1;
|
2004-12-04 18:41:22 +00:00
|
|
|
dy = y - y1;
|
|
|
|
|
2015-06-30 10:25:32 +02:00
|
|
|
if(pp->isPt()){
|
|
|
|
dx2 = int(pp->getScrX());
|
|
|
|
}else if(pp->isBranchEnd()) {
|
|
|
|
break;
|
|
|
|
}else if(pp->isStrokeEnd()) {
|
2007-02-08 07:04:54 +00:00
|
|
|
pp++;
|
2015-06-30 10:25:32 +02:00
|
|
|
dx2 = int(pp->getScrX()); // go on as graph can also be selected between strokes
|
2015-06-18 12:58:33 +02:00
|
|
|
if(pp->isBranchEnd()) 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
|
|
|
|
2015-06-30 10:25:32 +02:00
|
|
|
dy2 = int(pp->getScrY());
|
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()
|
|
|
|
{
|
2015-06-29 22:02:30 +02:00
|
|
|
Graph *pg = new Graph(diagram, Var);
|
2004-08-14 06:40:55 +00:00
|
|
|
|
|
|
|
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
|
|
|
|
2014-09-03 15:52:40 +02:00
|
|
|
foreach(Marker *pm, Markers)
|
2004-08-20 20:13:01 +00:00
|
|
|
pg->Markers.append(pm->sameNewOne(pg));
|
|
|
|
|
2004-08-14 06:40:55 +00:00
|
|
|
return pg;
|
|
|
|
}
|
2015-06-16 09:20:21 +02:00
|
|
|
|
2015-06-30 18:21:06 +02:00
|
|
|
/*!
|
|
|
|
* find a sample point close to VarPos, snap to it, and return data at VarPos
|
|
|
|
*/
|
|
|
|
std::pair<double,double> Graph::findSample(std::vector<double>& VarPos) const
|
2015-06-27 17:40:53 +02:00
|
|
|
{
|
|
|
|
DataX const* pD;
|
|
|
|
unsigned nVarPos=0;
|
|
|
|
unsigned n=0;
|
|
|
|
unsigned m=1;
|
|
|
|
|
|
|
|
for(unsigned ii=0; (pD=axis(ii)); ++ii) {
|
|
|
|
double* pp = pD->Points;
|
|
|
|
double v = VarPos[nVarPos];
|
2022-07-05 07:08:28 -04:00
|
|
|
for(unsigned i=pD->count; i>1; i--) { // find appropriate marker position
|
2015-06-27 17:40:53 +02:00
|
|
|
if(fabs(v-(*pp)) < fabs(v-(*(pp+1)))) break;
|
|
|
|
pp++;
|
|
|
|
n += m;
|
|
|
|
}
|
|
|
|
|
|
|
|
m *= pD->count;
|
|
|
|
VarPos[nVarPos++] = *pp;
|
|
|
|
}
|
|
|
|
|
2015-06-30 18:21:06 +02:00
|
|
|
return std::pair<double,double>(cPointsY[2*n], cPointsY[2*n+1]);
|
2015-06-27 17:40:53 +02:00
|
|
|
}
|
|
|
|
|
2015-06-18 12:58:33 +02:00
|
|
|
// -----------------------------------------------------------------------
|
|
|
|
// meaning of the values in a graph "Points" list
|
|
|
|
#define STROKEEND -2
|
|
|
|
#define BRANCHEND -10
|
|
|
|
#define GRAPHEND -100
|
|
|
|
// -----------------------------------------------------------------------
|
|
|
|
// screen points pseudo iterator implementation.
|
2015-06-18 13:03:43 +02:00
|
|
|
void Graph::ScrPt::setStrokeEnd()
|
|
|
|
{
|
2015-06-30 10:25:32 +02:00
|
|
|
ScrX = STROKEEND;
|
2015-06-18 13:03:43 +02:00
|
|
|
}
|
|
|
|
void Graph::ScrPt::setBranchEnd()
|
|
|
|
{
|
2015-06-30 10:25:32 +02:00
|
|
|
ScrX = BRANCHEND;
|
2015-06-18 13:03:43 +02:00
|
|
|
}
|
|
|
|
void Graph::ScrPt::setGraphEnd()
|
|
|
|
{
|
2015-06-30 10:25:32 +02:00
|
|
|
ScrX = GRAPHEND;
|
2015-06-18 13:03:43 +02:00
|
|
|
}
|
2015-06-30 10:25:32 +02:00
|
|
|
bool Graph::ScrPt::isPt() const{return ScrX>=0.;}
|
|
|
|
bool Graph::ScrPt::isStrokeEnd() const{return ScrX<=STROKEEND;}
|
|
|
|
bool Graph::ScrPt::isBranchEnd() const{return ScrX<=BRANCHEND;}
|
|
|
|
bool Graph::ScrPt::isGraphEnd() const{return ScrX<=GRAPHEND;}
|
2015-06-18 12:58:33 +02:00
|
|
|
|
2015-07-16 11:32:05 +02:00
|
|
|
/*!
|
|
|
|
* set screen coordinate for graph sampling point
|
|
|
|
* these must be nonnegative, but sometimes aren't,
|
|
|
|
* eg. between calcCoordinate and clip.
|
|
|
|
* (negative values are reserved for control.)
|
|
|
|
*/
|
2015-06-30 10:25:32 +02:00
|
|
|
void Graph::ScrPt::setScrX(float x)
|
2015-06-18 13:03:43 +02:00
|
|
|
{
|
2015-07-16 11:32:05 +02:00
|
|
|
if(x<0){
|
|
|
|
std::cerr << "dangerous: negative screen coordinate" << x;
|
|
|
|
}
|
|
|
|
if(ScrX<0){
|
|
|
|
std::cerr << "dangerous: (maybe) overwriting control token" << x;
|
2015-06-18 13:03:43 +02:00
|
|
|
}
|
2015-07-16 11:32:05 +02:00
|
|
|
ScrX = x;
|
2015-06-18 13:03:43 +02:00
|
|
|
}
|
2015-06-30 10:25:32 +02:00
|
|
|
void Graph::ScrPt::setScrY(float x)
|
|
|
|
{
|
2015-07-03 17:50:16 +02:00
|
|
|
if(x<0){ // need to investigate...
|
|
|
|
qDebug() << "setting negative screen coordinate" << x << "at" << ScrX;
|
|
|
|
}
|
2015-06-30 10:25:32 +02:00
|
|
|
ScrY = x;
|
|
|
|
}
|
|
|
|
void Graph::ScrPt::setScr(float x, float y)
|
|
|
|
{
|
|
|
|
setScrX(x);
|
|
|
|
setScrY(y);
|
|
|
|
}
|
|
|
|
void Graph::ScrPt::setIndep(double x)
|
|
|
|
{
|
|
|
|
assert(ScrX>=0);
|
|
|
|
indep = x;
|
|
|
|
}
|
|
|
|
void Graph::ScrPt::setDep(double x)
|
|
|
|
{
|
|
|
|
assert(ScrX>=0);
|
|
|
|
dep = x;
|
|
|
|
}
|
|
|
|
float Graph::ScrPt::getScrX() const
|
|
|
|
{
|
2015-07-16 11:32:05 +02:00
|
|
|
if(ScrX<0){
|
|
|
|
std::cerr << "dangerous: returning negative screen coordinate" << ScrX;
|
|
|
|
}
|
2015-06-30 10:25:32 +02:00
|
|
|
return ScrX;
|
|
|
|
}
|
|
|
|
float Graph::ScrPt::getScrY() const
|
|
|
|
{
|
|
|
|
return ScrY;
|
|
|
|
}
|
|
|
|
double Graph::ScrPt::getIndep() const
|
|
|
|
{
|
|
|
|
assert(ScrX>=0);
|
|
|
|
return indep;
|
|
|
|
}
|
|
|
|
double Graph::ScrPt::getDep() const
|
2015-06-18 13:03:43 +02:00
|
|
|
{
|
2015-06-30 10:25:32 +02:00
|
|
|
assert(ScrX>=0);
|
|
|
|
return dep;
|
2015-06-18 13:03:43 +02:00
|
|
|
}
|
|
|
|
|
2015-06-16 09:20:21 +02:00
|
|
|
// vim:ts=8:sw=2:et
|