2018-06-14 07:25:54 +00:00
/* gGraph Implemntation
2014-04-23 17:20:38 +00:00
*
2021-11-02 20:34:12 +00:00
* Copyright ( c ) 2019 - 2022 The OSCAR Team
2018-04-02 02:08:32 +00:00
* Copyright ( c ) 2011 - 2018 Mark Watkins < mark @ jedimark . net >
2014-04-23 17:20:38 +00:00
*
* This file is subject to the terms and conditions of the GNU General Public
2018-06-04 20:48:38 +00:00
* License . See the file COPYING in the main directory of the source code
* for more details . */
2014-04-24 09:44:15 +00:00
2014-04-23 17:20:38 +00:00
# include "Graphs/gGraph.h"
# include <QLabel>
# include <QTimer>
2014-04-23 22:26:49 +00:00
# include <cmath>
2014-07-16 17:12:52 +00:00
# include <exception>
2014-04-23 17:20:38 +00:00
# include "mainwindow.h"
# include "Graphs/gGraphView.h"
# include "Graphs/layer.h"
# include "SleepLib/profiles.h"
extern MainWindow * mainwin ;
// Graph globals.
QFont * defaultfont = nullptr ;
QFont * mediumfont = nullptr ;
QFont * bigfont = nullptr ;
QHash < QString , QImage * > images ;
static bool globalsInitialized = false ;
// Graph constants.
static const double zoom_hard_limit = 500.0 ;
2014-04-24 09:44:15 +00:00
2016-03-07 13:54:14 +00:00
// Calculate Catmull-Rom Spline of given 4 samples, with t between 0-1;
float CatmullRomSpline ( float p0 , float p1 , float p2 , float p3 , float t )
{
float t2 = t * t ;
float t3 = t2 * t ;
return ( float ) 0.5 * ( ( 2 * p1 ) +
( - p0 + p2 ) * t +
( 2 * p0 - 5 * p1 + 4 * p2 - p3 ) * t2 +
( - p0 + 3 * p1 - 3 * p2 + p3 ) * t3 ) ;
}
2014-04-23 17:20:38 +00:00
// Must be called from a thread inside the application.
bool InitGraphGlobals ( )
{
if ( globalsInitialized ) {
return true ;
}
2019-03-25 23:36:41 +00:00
if ( ! p_pref - > contains ( " Fonts_Graph_Name " ) ) {
( * p_pref ) [ " Fonts_Graph_Name " ] = " Sans Serif " ;
( * p_pref ) [ " Fonts_Graph_Size " ] = 10 ;
( * p_pref ) [ " Fonts_Graph_Bold " ] = false ;
( * p_pref ) [ " Fonts_Graph_Italic " ] = false ;
2014-04-23 17:20:38 +00:00
}
2019-03-25 23:36:41 +00:00
if ( ! p_pref - > contains ( " Fonts_Title_Name " ) ) {
( * p_pref ) [ " Fonts_Title_Name " ] = " Sans Serif " ;
( * p_pref ) [ " Fonts_Title_Size " ] = 12 ;
( * p_pref ) [ " Fonts_Title_Bold " ] = true ;
( * p_pref ) [ " Fonts_Title_Italic " ] = false ;
2014-04-23 17:20:38 +00:00
}
2019-03-25 23:36:41 +00:00
if ( ! p_pref - > contains ( " Fonts_Big_Name " ) ) {
( * p_pref ) [ " Fonts_Big_Name " ] = " Serif " ;
( * p_pref ) [ " Fonts_Big_Size " ] = 35 ;
( * p_pref ) [ " Fonts_Big_Bold " ] = false ;
( * p_pref ) [ " Fonts_Big_Italic " ] = false ;
2014-04-23 17:20:38 +00:00
}
2019-03-25 23:36:41 +00:00
defaultfont = new QFont ( ( * p_pref ) [ " Fonts_Graph_Name " ] . toString ( ) ,
( * p_pref ) [ " Fonts_Graph_Size " ] . toInt ( ) ,
( * p_pref ) [ " Fonts_Graph_Bold " ] . toBool ( ) ? QFont : : Bold : QFont : : Normal ,
( * p_pref ) [ " Fonts_Graph_Italic " ] . toBool ( )
2014-04-23 17:20:38 +00:00
) ;
2019-03-25 23:36:41 +00:00
mediumfont = new QFont ( ( * p_pref ) [ " Fonts_Title_Name " ] . toString ( ) ,
( * p_pref ) [ " Fonts_Title_Size " ] . toInt ( ) ,
( * p_pref ) [ " Fonts_Title_Bold " ] . toBool ( ) ? QFont : : Bold : QFont : : Normal ,
( * p_pref ) [ " Fonts_Title_Italic " ] . toBool ( )
2014-04-23 17:20:38 +00:00
) ;
2019-03-25 23:36:41 +00:00
bigfont = new QFont ( ( * p_pref ) [ " Fonts_Big_Name " ] . toString ( ) ,
( * p_pref ) [ " Fonts_Big_Size " ] . toInt ( ) ,
( * p_pref ) [ " Fonts_Big_Bold " ] . toBool ( ) ? QFont : : Bold : QFont : : Normal ,
( * p_pref ) [ " Fonts_Big_Italic " ] . toBool ( )
2014-04-23 17:20:38 +00:00
) ;
defaultfont - > setStyleHint ( QFont : : AnyStyle , QFont : : OpenGLCompatible ) ;
mediumfont - > setStyleHint ( QFont : : AnyStyle , QFont : : OpenGLCompatible ) ;
bigfont - > setStyleHint ( QFont : : AnyStyle , QFont : : OpenGLCompatible ) ;
//images["mask"] = new QImage(":/icons/mask.png");
images [ " oximeter " ] = new QImage ( " :/icons/cubeoximeter.png " ) ;
images [ " smiley " ] = new QImage ( " :/icons/smileyface.png " ) ;
//images["sad"] = new QImage(":/icons/sadface.png");
2019-04-16 17:12:26 +00:00
images [ " logo " ] = new QImage ( " :/icons/logo-lm.png " ) ;
2014-04-23 17:20:38 +00:00
images [ " brick " ] = new QImage ( " :/icons/brick.png " ) ;
images [ " nographs " ] = new QImage ( " :/icons/nographs.png " ) ;
images [ " nodata " ] = new QImage ( " :/icons/nodata.png " ) ;
globalsInitialized = true ;
return true ;
}
void DestroyGraphGlobals ( )
{
if ( ! globalsInitialized ) {
return ;
}
delete defaultfont ;
delete bigfont ;
delete mediumfont ;
2018-05-05 21:58:11 +00:00
for ( auto & image : images ) {
delete image ;
2014-04-23 17:20:38 +00:00
}
globalsInitialized = false ;
}
2014-04-24 09:44:15 +00:00
2014-07-16 17:12:52 +00:00
gGraph : : gGraph ( QString name , gGraphView * graphview , QString title , QString units , int height , short group )
: m_name ( name ) ,
m_graphview ( graphview ) ,
2014-04-23 17:20:38 +00:00
m_title ( title ) ,
m_units ( units ) ,
m_visible ( true )
{
2014-08-09 17:12:37 +00:00
if ( height = = 0 ) {
2018-04-22 12:06:48 +00:00
height = AppSetting - > graphHeight ( ) ;
2018-03-28 06:22:42 +00:00
Q_UNUSED ( height )
2014-08-09 17:12:37 +00:00
}
2018-03-28 06:22:42 +00:00
2014-08-27 13:42:10 +00:00
if ( graphview & & graphview - > contains ( name ) ) {
2014-07-16 17:12:52 +00:00
qDebug ( ) < < " Trying to duplicate " < < name < < " when a graph with the same name already exists " ;
name + = " -1 " ;
}
2014-04-23 17:20:38 +00:00
m_min_height = 60 ;
m_width = 0 ;
m_layers . clear ( ) ;
2014-08-28 08:01:25 +00:00
m_snapshot = false ;
2014-04-23 17:20:38 +00:00
f_miny = f_maxy = 0 ;
rmin_x = rmin_y = 0 ;
rmax_x = rmax_y = 0 ;
max_x = max_y = 0 ;
min_x = min_y = 0 ;
rec_miny = rec_maxy = 0 ;
rphysmax_y = rphysmin_y = 0 ;
m_zoomY = 0 ;
2014-07-19 12:51:17 +00:00
m_selectedDuration = 0 ;
2014-04-23 17:20:38 +00:00
if ( graphview ) {
graphview - > addGraph ( this , group ) ;
timer = new QTimer ( graphview ) ;
connect ( timer , SIGNAL ( timeout ( ) ) , SLOT ( Timeout ( ) ) ) ;
} else {
2014-08-28 08:01:25 +00:00
timer = new QTimer ( ) ;
connect ( timer , SIGNAL ( timeout ( ) ) , SLOT ( Timeout ( ) ) ) ;
2014-08-27 13:42:10 +00:00
// know what I'm doing now.. ;)
// qWarning() << "gGraph created without a gGraphView container.. Naughty programmer!! Bad!!!";
2014-04-23 17:20:38 +00:00
}
m_margintop = 14 ;
m_marginbottom = 5 ;
m_marginleft = 0 ;
m_marginright = 15 ;
m_selecting_area = m_blockzoom = false ;
m_pinned = false ;
m_lastx23 = 0 ;
invalidate_yAxisImage = true ;
invalidate_xAxisImage = true ;
2014-08-12 06:43:10 +00:00
m_block_select = false ;
2014-04-23 17:20:38 +00:00
m_enforceMinY = m_enforceMaxY = false ;
m_showTitle = true ;
m_printing = false ;
2014-10-03 07:09:35 +00:00
left = right = top = bottom = 0 ;
2014-04-23 17:20:38 +00:00
}
gGraph : : ~ gGraph ( )
{
2018-05-05 21:58:11 +00:00
for ( auto & layer : m_layers ) {
if ( layer - > unref ( ) ) {
delete layer ;
2014-04-23 17:20:38 +00:00
}
}
m_layers . clear ( ) ;
2014-08-27 13:42:10 +00:00
if ( timer ) {
timer - > stop ( ) ;
disconnect ( timer , 0 , 0 , 0 ) ;
delete timer ;
}
2014-04-23 17:20:38 +00:00
}
void gGraph : : Trigger ( int ms )
{
if ( timer - > isActive ( ) ) { timer - > stop ( ) ; }
timer - > setSingleShot ( true ) ;
timer - > start ( ms ) ;
}
void gGraph : : Timeout ( )
{
deselect ( ) ;
m_graphview - > timedRedraw ( 0 ) ;
}
void gGraph : : deselect ( )
{
2018-05-05 21:58:11 +00:00
for ( auto & layer : m_layers ) {
layer - > deselect ( ) ;
2014-04-23 17:20:38 +00:00
}
}
bool gGraph : : isSelected ( )
{
bool res = false ;
2018-05-05 21:58:11 +00:00
for ( const auto & layer : m_layers ) {
res = layer - > isSelected ( ) ;
2014-04-23 17:20:38 +00:00
if ( res ) { break ; }
}
return res ;
}
bool gGraph : : isEmpty ( )
{
bool empty = true ;
2018-05-05 21:58:11 +00:00
for ( const auto & layer : m_layers ) {
if ( ! layer - > isEmpty ( ) ) {
2014-04-23 17:20:38 +00:00
empty = false ;
break ;
}
}
return empty ;
}
float gGraph : : printScaleX ( ) { return m_graphview - > printScaleX ( ) ; }
float gGraph : : printScaleY ( ) { return m_graphview - > printScaleY ( ) ; }
2014-05-08 04:07:23 +00:00
//void gGraph::drawGLBuf()
//{
2014-04-23 17:20:38 +00:00
2014-05-08 04:07:23 +00:00
// float linesize = 1;
2014-04-23 17:20:38 +00:00
2014-05-08 04:07:23 +00:00
// if (m_printing) { linesize = 4; } //ceil(m_graphview->printScaleY());
2014-04-23 17:20:38 +00:00
2014-05-08 04:07:23 +00:00
// for (int i = 0; i < m_layers.size(); i++) {
// m_layers[i]->drawGLBuf(linesize);
// }
2014-04-23 17:20:38 +00:00
2014-05-08 04:07:23 +00:00
//}
2019-08-22 09:31:33 +00:00
2014-04-23 17:20:38 +00:00
void gGraph : : setDay ( Day * day )
{
2014-08-28 08:01:25 +00:00
// Don't update for snapshots..
if ( m_snapshot ) return ;
2014-04-23 17:20:38 +00:00
m_day = day ;
2018-05-05 21:58:11 +00:00
for ( auto & layer : m_layers ) {
layer - > SetDay ( day ) ;
2014-04-23 17:20:38 +00:00
}
2014-08-27 09:00:55 +00:00
rmin_y = rmax_y = 0 ;
2021-03-28 03:20:19 +00:00
// This resets weight and bmi overview graphs to full date range when they are changed.
// is it required ever?
// ResetBounds();
2014-04-23 17:20:38 +00:00
}
void gGraph : : setZoomY ( short zoom )
{
m_zoomY = zoom ;
2014-08-26 17:26:13 +00:00
timedRedraw ( 0 ) ;
2014-04-23 17:20:38 +00:00
}
2014-05-10 00:50:35 +00:00
void gGraph : : renderText ( QString text , int x , int y , float angle , QColor color , QFont * font , bool antialias )
2014-04-23 17:20:38 +00:00
{
m_graphview - > AddTextQue ( text , x , y , angle , color , font , antialias ) ;
}
2014-08-17 12:56:05 +00:00
void gGraph : : renderText ( QString text , QRectF rect , quint32 flags , float angle , QColor color , QFont * font , bool antialias )
2014-08-12 10:44:05 +00:00
{
m_graphview - > AddTextQue ( text , rect , flags , angle , color , font , antialias ) ;
}
2014-05-13 17:47:33 +00:00
void gGraph : : paint ( QPainter & painter , const QRegion & region )
2014-04-23 17:20:38 +00:00
{
2014-05-13 17:47:33 +00:00
m_rect = region . boundingRect ( ) ;
int originX = m_rect . left ( ) ;
int originY = m_rect . top ( ) ;
int width = m_rect . width ( ) ;
int height = m_rect . height ( ) ;
2014-04-23 17:20:38 +00:00
int fw , font_height ;
GetTextExtent ( " Wg@ " , fw , font_height ) ;
2014-08-29 04:02:16 +00:00
if ( m_margintop > 0 ) {
m_margintop = font_height + ( 2 * printScaleY ( ) ) ;
}
2014-04-23 17:20:38 +00:00
//m_marginbottom=5;
2014-08-27 17:30:53 +00:00
left = marginLeft ( ) * printScaleX ( ) , right = marginRight ( ) * printScaleX ( ) , top = marginTop ( ) , bottom = marginBottom ( ) * printScaleY ( ) ;
2014-10-02 10:02:33 +00:00
//int x;
int y ;
2014-04-23 17:20:38 +00:00
if ( m_showTitle ) {
int title_x , yh ;
2014-08-27 17:30:53 +00:00
painter . setFont ( * mediumfont ) ;
2014-04-23 17:20:38 +00:00
QFontMetrics fm ( * mediumfont ) ;
yh = fm . height ( ) ;
//GetTextExtent("Wy@",x,yh,mediumfont); // This gets a better consistent height. should be cached.
y = yh ;
2014-10-02 10:02:33 +00:00
//x = fm.width(title());
2014-04-23 17:20:38 +00:00
//GetTextExtent(title(),x,y,mediumfont);
2014-08-27 17:30:53 +00:00
title_x = float ( yh ) ;
2014-04-23 17:20:38 +00:00
2014-05-07 19:52:59 +00:00
QString & txt = title ( ) ;
2014-08-28 08:01:25 +00:00
graphView ( ) - > AddTextQue ( txt , marginLeft ( ) + title_x + 8 * printScaleX ( ) , originY + height / 2 - y / 2 , 90 , Qt : : black , mediumfont ) ;
2014-08-17 12:56:05 +00:00
2014-08-27 17:30:53 +00:00
left + = graphView ( ) - > titleWidth * printScaleX ( ) ;
2014-04-23 17:20:38 +00:00
} else { left = 0 ; }
2014-08-27 13:42:10 +00:00
2014-08-28 08:01:25 +00:00
if ( m_snapshot ) {
2014-08-27 13:42:10 +00:00
QLinearGradient linearGrad ( QPointF ( 100 , 100 ) , QPointF ( width / 2 , 100 ) ) ;
2014-08-28 15:11:34 +00:00
linearGrad . setColorAt ( 0 , QColor ( 255 , 150 , 150 , 40 ) ) ;
2014-08-27 13:42:10 +00:00
linearGrad . setColorAt ( 1 , QColor ( 255 , 255 , 255 , 20 ) ) ;
painter . fillRect ( m_rect , QBrush ( linearGrad ) ) ;
painter . setFont ( * defaultfont ) ;
painter . setPen ( QColor ( 0 , 0 , 0 , 255 ) ) ;
QString t = name ( ) . section ( " ; " , - 1 ) ;
2014-08-29 04:02:16 +00:00
QString txt = QObject : : tr ( " Snapshot %1 " ) . arg ( t ) ;
QRectF rec = QRect ( m_rect . left ( ) , m_rect . top ( ) + 6 * printScaleY ( ) , m_rect . width ( ) , 0 ) ;
rec = painter . boundingRect ( rec , Qt : : AlignCenter , txt ) ;
painter . drawText ( rec , Qt : : AlignCenter , txt ) ;
m_margintop + = rec . height ( ) ;
top = m_margintop ;
2014-08-27 13:42:10 +00:00
}
2014-04-23 17:20:38 +00:00
# ifdef DEBUG_LAYOUT
QColor col = Qt : : red ;
2014-05-07 19:52:59 +00:00
painter . setPen ( col ) ;
painter . drawLine ( 0 , originY , 0 , originY + height ) ;
painter . drawLine ( left , originY , left , originY + height ) ;
2014-04-23 17:20:38 +00:00
# endif
int tmp ;
2014-08-27 13:42:10 +00:00
// left = 0;
2014-04-23 17:20:38 +00:00
2018-05-05 21:58:11 +00:00
for ( const auto & layer : m_layers ) {
if ( ! layer - > visible ( ) ) { continue ; }
2014-04-23 17:20:38 +00:00
2018-05-05 21:58:11 +00:00
tmp = layer - > minimumHeight ( ) ; // * m_graphview->printScaleY();
2014-04-23 17:20:38 +00:00
2018-05-05 21:58:11 +00:00
if ( layer - > position ( ) = = LayerTop ) { top + = tmp ; }
if ( layer - > position ( ) = = LayerBottom ) { bottom + = tmp * printScaleY ( ) ; }
2014-04-23 17:20:38 +00:00
}
2018-05-05 21:58:11 +00:00
for ( const auto & layer : m_layers ) {
if ( ! layer - > visible ( ) ) { continue ; }
2014-04-23 17:20:38 +00:00
2018-05-05 21:58:11 +00:00
tmp = layer - > minimumWidth ( ) ;
2014-08-27 13:42:10 +00:00
tmp * = m_graphview - > printScaleX ( ) ;
tmp * = m_graphview - > devicePixelRatio ( ) ;
2014-04-23 17:20:38 +00:00
2018-05-05 21:58:11 +00:00
if ( layer - > position ( ) = = LayerLeft ) {
2014-05-13 17:47:33 +00:00
QRect rect ( originX + left , originY + top , tmp , height - top - bottom ) ;
2018-05-05 21:58:11 +00:00
layer - > m_rect = rect ;
// layer->paint(painter, *this, QRegion(rect));
2014-04-23 17:20:38 +00:00
left + = tmp ;
# ifdef DEBUG_LAYOUT
2014-05-07 19:52:59 +00:00
QColor col = Qt : : red ;
painter . setPen ( col ) ;
painter . drawLine ( originX + left - 1 , originY , originX + left - 1 , originY + height ) ;
2014-04-23 17:20:38 +00:00
# endif
}
2018-05-05 21:58:11 +00:00
if ( layer - > position ( ) = = LayerRight ) {
2014-04-23 17:20:38 +00:00
right + = tmp ;
2014-05-13 17:47:33 +00:00
QRect rect ( originX + width - right , originY + top , tmp , height - top - bottom ) ;
2018-05-05 21:58:11 +00:00
layer - > m_rect = rect ;
//layer->paint(painter, *this, QRegion(rect));
2014-04-23 17:20:38 +00:00
# ifdef DEBUG_LAYOUT
2014-05-07 19:52:59 +00:00
QColor col = Qt : : red ;
painter . setPen ( col ) ;
painter . drawLine ( originX + width - right , originY , originX + width - right , originY + height ) ;
2014-04-23 17:20:38 +00:00
# endif
}
}
2014-08-27 17:30:53 +00:00
bottom = marginBottom ( ) * printScaleY ( ) ;
2014-04-23 17:20:38 +00:00
top = marginTop ( ) ;
2018-05-05 21:58:11 +00:00
for ( const auto & layer : m_layers ) {
if ( ! layer - > visible ( ) ) { continue ; }
2014-04-23 17:20:38 +00:00
2018-05-05 21:58:11 +00:00
tmp = layer - > minimumHeight ( ) ;
2014-04-23 17:20:38 +00:00
2018-05-05 21:58:11 +00:00
if ( layer - > position ( ) = = LayerTop ) {
2014-05-13 17:47:33 +00:00
QRect rect ( originX + left , originY + top , width - left - right , tmp ) ;
2018-05-05 21:58:11 +00:00
layer - > m_rect = rect ;
layer - > paint ( painter , * this , QRegion ( rect ) ) ;
2014-04-23 17:20:38 +00:00
top + = tmp ;
}
2018-05-05 21:58:11 +00:00
if ( layer - > position ( ) = = LayerBottom ) {
2014-08-27 17:30:53 +00:00
bottom + = tmp * printScaleY ( ) ;
2014-05-13 17:47:33 +00:00
QRect rect ( originX + left , originY + height - bottom , width - left - right , tmp ) ;
2018-05-05 21:58:11 +00:00
layer - > m_rect = rect ;
layer - > paint ( painter , * this , QRegion ( rect ) ) ;
2014-04-23 17:20:38 +00:00
}
}
if ( isPinned ( ) ) {
// Fill the background on pinned graphs
2014-05-07 19:52:59 +00:00
painter . fillRect ( originX + left , originY + top , width - right , height - bottom - top , QBrush ( QColor ( Qt : : white ) ) ) ;
2014-04-23 17:20:38 +00:00
}
2018-05-05 21:58:11 +00:00
for ( const auto & layer : m_layers ) {
if ( ! layer - > visible ( ) ) { continue ; }
2014-04-23 17:20:38 +00:00
2018-05-05 21:58:11 +00:00
if ( layer - > position ( ) = = LayerCenter ) {
2014-05-13 17:47:33 +00:00
QRect rect ( originX + left , originY + top , width - left - right , height - top - bottom ) ;
2018-05-05 21:58:11 +00:00
layer - > m_rect = rect ;
layer - > paint ( painter , * this , QRegion ( rect ) ) ;
2014-04-23 17:20:38 +00:00
}
}
2014-09-13 11:34:18 +00:00
// Draw anything like the YAxis labels afterwards, in case the graph scale was updated during draw
2018-05-05 21:58:11 +00:00
for ( const auto & layer : m_layers ) {
if ( ! layer - > visible ( ) ) { continue ; }
if ( ( layer - > position ( ) = = LayerLeft ) | | ( layer - > position ( ) = = LayerRight ) ) {
layer - > paint ( painter , * this , QRegion ( layer - > m_rect ) ) ;
2014-09-13 11:34:18 +00:00
}
}
2014-04-23 17:20:38 +00:00
if ( m_selection . width ( ) > 0 & & m_selecting_area ) {
QColor col ( 128 , 128 , 255 , 128 ) ;
2014-05-07 19:52:59 +00:00
painter . fillRect ( originX + m_selection . x ( ) , originY + top , m_selection . width ( ) , height - bottom - top , QBrush ( col ) ) ;
// quads()->add(originX + m_selection.x(), originY + top,
// originX + m_selection.x() + m_selection.width(), originY + top, col.rgba());
// quads()->add(originX + m_selection.x() + m_selection.width(), originY + height - bottom,
// originX + m_selection.x(), originY + height - bottom, col.rgba());
2014-04-23 17:20:38 +00:00
}
2014-08-17 12:56:05 +00:00
2014-09-13 11:34:18 +00:00
if ( isPinned ( ) & & ! printing ( ) ) {
2014-08-17 12:56:05 +00:00
painter . drawPixmap ( - 5 , originY - 10 , m_graphview - > pin_icon ) ;
}
2014-04-23 17:20:38 +00:00
}
QPixmap gGraph : : renderPixmap ( int w , int h , bool printing )
{
QFont * _defaultfont = defaultfont ;
QFont * _mediumfont = mediumfont ;
QFont * _bigfont = bigfont ;
QFont fa = * defaultfont ;
QFont fb = * mediumfont ;
QFont fc = * bigfont ;
m_printing = printing ;
if ( printing ) {
2014-08-17 12:56:05 +00:00
fa . setPixelSize ( 28 ) ;
fb . setPixelSize ( 32 ) ;
fc . setPixelSize ( 70 ) ;
2014-10-02 07:56:57 +00:00
graphView ( ) - > setPrintScaleX ( 2.5f ) ;
graphView ( ) - > setPrintScaleY ( 2.2f ) ;
2014-04-23 17:20:38 +00:00
} else {
2014-10-02 07:56:57 +00:00
graphView ( ) - > setPrintScaleX ( 1.0f ) ;
graphView ( ) - > setPrintScaleY ( 1.0f ) ;
2014-04-23 17:20:38 +00:00
}
defaultfont = & fa ;
mediumfont = & fb ;
bigfont = & fc ;
2014-05-07 19:52:59 +00:00
QPixmap pm ( w , h ) ;
2014-04-23 17:20:38 +00:00
2014-08-29 04:02:16 +00:00
2018-04-22 12:06:48 +00:00
bool pixcaching = AppSetting - > usePixmapCaching ( ) ;
2014-08-17 12:56:05 +00:00
graphView ( ) - > setUsePixmapCache ( false ) ;
2018-04-22 12:06:48 +00:00
AppSetting - > setUsePixmapCaching ( false ) ;
2014-05-07 19:52:59 +00:00
QPainter painter ( & pm ) ;
painter . fillRect ( 0 , 0 , w , h , QBrush ( QColor ( Qt : : white ) ) ) ;
2014-08-17 12:56:05 +00:00
QRegion region ( 0 , 0 , w , h ) ;
paint ( painter , region ) ;
DrawTextQue ( painter ) ;
2014-05-07 19:52:59 +00:00
painter . end ( ) ;
2014-04-23 17:20:38 +00:00
2014-08-29 04:02:16 +00:00
graphView ( ) - > setUsePixmapCache ( pixcaching ) ;
2018-04-22 12:06:48 +00:00
AppSetting - > setUsePixmapCaching ( pixcaching ) ;
2014-08-17 12:56:05 +00:00
graphView ( ) - > setPrintScaleX ( 1 ) ;
graphView ( ) - > setPrintScaleY ( 1 ) ;
2014-04-23 17:20:38 +00:00
defaultfont = _defaultfont ;
mediumfont = _mediumfont ;
bigfont = _bigfont ;
m_printing = false ;
return pm ;
}
2019-08-22 09:31:33 +00:00
// Sets a new Min & Max X dates for clipping data (refresh done by caller)
2014-04-23 17:20:38 +00:00
void gGraph : : SetXBounds ( qint64 minx , qint64 maxx )
{
invalidate_xAxisImage = true ;
min_x = minx ;
max_x = maxx ;
//repaint();
//m_graphview->redraw();
}
int gGraph : : flipY ( int y )
{
return m_graphview - > height ( ) - y ;
}
void gGraph : : ResetBounds ( )
{
2014-08-28 08:01:25 +00:00
if ( m_snapshot ) return ;
2014-04-23 17:20:38 +00:00
invalidate_xAxisImage = true ;
min_x = MinX ( ) ;
max_x = MaxX ( ) ;
min_y = MinY ( ) ;
max_y = MaxY ( ) ;
}
2014-08-06 05:30:15 +00:00
void gGraph : : ToolTip ( QString text , int x , int y , ToolTipAlignment align , int timeout )
2014-04-23 17:20:38 +00:00
{
if ( timeout < = 0 ) {
2018-04-22 12:06:48 +00:00
timeout = AppSetting - > tooltipTimeout ( ) ;
2014-04-23 17:20:38 +00:00
}
2014-08-06 05:30:15 +00:00
m_graphview - > m_tooltip - > display ( text , x , y , align , timeout ) ;
2014-04-23 17:20:38 +00:00
}
// YAxis Autoscaling code
void gGraph : : roundY ( EventDataType & miny , EventDataType & maxy )
{
2014-05-20 14:52:29 +00:00
2014-08-27 09:00:55 +00:00
if ( zoomY ( ) = = 2 ) {
2014-08-27 04:36:40 +00:00
miny = rec_miny ;
maxy = rec_maxy ;
2014-08-27 09:00:55 +00:00
if ( maxy > miny ) return ;
} else if ( zoomY ( ) = = 1 ) {
miny = physMinY ( ) ;
maxy = physMaxY ( ) ;
if ( maxy > miny ) return ;
2014-05-20 14:52:29 +00:00
}
2014-08-27 09:00:55 +00:00
miny = MinY ( ) ;
maxy = MaxY ( ) ;
2014-05-20 14:52:29 +00:00
2014-04-23 17:20:38 +00:00
int m , t ;
bool ymin_good = false , ymax_good = false ;
2014-05-20 11:51:47 +00:00
// Have no minx/miny reference, have to create one
2014-04-23 17:20:38 +00:00
if ( maxy = = miny ) {
m = ceil ( maxy / 2.0 ) ;
t = m * 2 ;
if ( maxy = = t ) {
t + = 2 ;
}
if ( ! ymax_good ) {
maxy = t ;
}
m = floor ( miny / 2.0 ) ;
t = m * 2 ;
if ( miny = = t ) {
t - = 2 ;
}
if ( miny > = 0 & & t < 0 ) {
t = 0 ;
}
if ( ! ymin_good ) {
miny = t ;
}
2014-05-20 11:51:47 +00:00
if ( miny < 0 ) {
EventDataType tmp = qMax ( qAbs ( miny ) , qAbs ( maxy ) ) ;
maxy = tmp ;
miny = - tmp ;
}
2014-04-23 17:20:38 +00:00
return ;
}
if ( maxy > = 400 ) {
m = ceil ( maxy / 50.0 ) ;
t = m * 50 ;
if ( ! ymax_good ) {
maxy = t ;
}
m = floor ( miny / 50.0 ) ;
if ( ! ymin_good ) {
miny = m * 50 ;
}
2014-08-27 09:00:55 +00:00
} else if ( maxy > = 30 ) {
2014-04-23 17:20:38 +00:00
m = ceil ( maxy / 5.0 ) ;
t = m * 5 ;
if ( ! ymax_good ) {
maxy = t ;
}
m = floor ( miny / 5.0 ) ;
if ( ! ymin_good ) {
miny = m * 5 ;
}
} else {
if ( maxy = = miny & & maxy = = 0 ) {
maxy = 0.5 ;
} else {
//maxy*=4.0;
//miny*=4.0;
if ( ! ymax_good ) {
maxy = ceil ( maxy ) ;
}
if ( ! ymin_good ) {
miny = floor ( miny ) ;
}
//maxy/=4.0;
//miny/=4.0;
}
}
2020-01-30 18:45:55 +00:00
// Make the range symmetrical if there are both positive and negative values.
if ( miny < 0 & & maxy > 0 ) {
2014-05-20 11:51:47 +00:00
EventDataType tmp = qMax ( qAbs ( miny ) , qAbs ( maxy ) ) ;
maxy = tmp ;
miny = - tmp ;
}
2014-04-23 17:20:38 +00:00
//if (m_enforceMinY) { miny=f_miny; }
//if (m_enforceMaxY) { maxy=f_maxy; }
}
2014-08-09 17:34:00 +00:00
void gGraph : : AddLayer ( Layer * l , LayerPosition position , short width , short height , short order , bool movable , short x , short y )
2014-04-23 17:20:38 +00:00
{
l - > setLayout ( position , width , height , order ) ;
l - > setMovable ( movable ) ;
l - > setPos ( x , y ) ;
l - > addref ( ) ;
m_layers . push_back ( l ) ;
}
2014-09-14 15:29:07 +00:00
void gGraph : : dataChanged ( )
{
2018-05-05 21:58:11 +00:00
for ( auto & layer : m_layers ) {
layer - > dataChanged ( ) ;
2014-09-14 15:29:07 +00:00
}
}
2014-04-23 17:20:38 +00:00
void gGraph : : redraw ( )
{
m_graphview - > redraw ( ) ;
}
void gGraph : : timedRedraw ( int ms )
{
m_graphview - > timedRedraw ( ms ) ;
}
2014-08-06 03:30:47 +00:00
double gGraph : : currentTime ( ) const
2014-07-21 07:14:02 +00:00
{
return m_graphview - > currentTime ( ) ;
}
2014-08-07 17:20:52 +00:00
double gGraph : : screenToTime ( int xpos )
{
double w = m_rect . width ( ) - left - right ;
double xx = m_blockzoom ? rmax_x - rmin_x : max_x - min_x ;
double xmult = xx / w ;
double x = xpos - m_rect . left ( ) - left ;
double res = xmult * x ;
res + = m_blockzoom ? rmin_x : min_x ;
return res ;
}
2014-07-21 07:14:02 +00:00
2014-04-23 17:20:38 +00:00
void gGraph : : mouseMoveEvent ( QMouseEvent * event )
{
2020-07-07 18:07:44 +00:00
// qDebug() << m_title << "Move" << event->pos() << m_graphview->pointClicked();
2014-07-25 07:53:48 +00:00
if ( m_rect . width ( ) = = 0 ) return ;
2014-04-23 17:20:38 +00:00
int y = event - > y ( ) ;
int x = event - > x ( ) ;
2014-10-02 10:02:33 +00:00
//bool doredraw = false;
2014-04-23 17:20:38 +00:00
2014-08-29 04:02:16 +00:00
timedRedraw ( 0 ) ;
2014-08-28 04:02:22 +00:00
2018-05-05 21:58:11 +00:00
for ( const auto & layer : m_layers ) {
if ( layer - > m_rect . contains ( x , y ) )
if ( layer - > mouseMoveEvent ( event , this ) ) {
2014-08-12 10:44:05 +00:00
return ;
2014-04-23 17:20:38 +00:00
}
}
y - = m_rect . top ( ) ;
x - = m_rect . left ( ) ;
int x2 = m_graphview - > pointClicked ( ) . x ( ) - m_rect . left ( ) ;
2014-07-21 09:39:26 +00:00
int w = m_rect . width ( ) - left - right ;
2018-06-12 16:32:18 +00:00
double xx ;
double xmult ;
2014-04-23 17:20:38 +00:00
2014-07-21 07:14:02 +00:00
{
2014-08-06 03:33:20 +00:00
xmult = ( m_blockzoom ? double ( rmax_x - rmin_x ) : double ( max_x - min_x ) ) / double ( w ) ;
2014-07-21 07:14:02 +00:00
double a = x ;
if ( a < left ) a = left ;
if ( a > left + w ) a = left + w ;
a - = left ;
a * = xmult ;
a + = m_blockzoom ? rmin_x : min_x ;
2014-07-25 07:53:48 +00:00
m_currentTime = a ;
2014-07-21 07:14:02 +00:00
m_graphview - > setCurrentTime ( a ) ;
}
2014-04-23 17:20:38 +00:00
if ( m_graphview - > m_selected_graph = = this ) { // Left Mouse button dragging
if ( event - > buttons ( ) & Qt : : LeftButton ) {
2014-07-19 12:51:17 +00:00
2014-04-23 17:20:38 +00:00
//qDebug() << m_title << "Moved" << x << y << left << right << top << bottom << m_width << h;
int a1 = MIN ( x , x2 ) ;
int a2 = MAX ( x , x2 ) ;
if ( a1 < left ) { a1 = left ; }
if ( a2 > left + w ) { a2 = left + w ; }
m_selecting_area = true ;
m_selection = QRect ( a1 - 1 , 0 , a2 - a1 , m_rect . height ( ) ) ;
double w2 = m_rect . width ( ) - right - left ;
if ( m_blockzoom ) {
xmult = ( rmax_x - rmin_x ) / w2 ;
} else {
xmult = ( max_x - min_x ) / w2 ;
}
qint64 a = double ( a2 - a1 ) * xmult ;
2014-07-19 12:51:17 +00:00
m_selectedDuration = a ;
2014-04-23 17:20:38 +00:00
float d = double ( a ) / 86400000.0 ;
int h = a / 3600000 ;
int m = ( a / 60000 ) % 60 ;
int s = ( a / 1000 ) % 60 ;
int ms ( a % 1000 ) ;
if ( d > 1 ) {
2018-06-14 07:25:54 +00:00
m_selDurString = tr ( " %1 days " ) . arg ( floor ( d ) ) ;
2014-04-23 17:20:38 +00:00
} else {
2014-07-19 12:51:17 +00:00
m_selDurString . sprintf ( " %02i:%02i:%02i:%03i " , h , m , s , ms ) ;
2014-04-23 17:20:38 +00:00
}
2014-08-11 01:20:09 +00:00
ToolTipAlignment align = x > = x2 ? TT_AlignLeft : TT_AlignRight ;
int offset = ( x > = x2 ) ? 20 : - 20 ;
ToolTip ( m_selDurString , m_rect . left ( ) + x + offset , m_rect . top ( ) + y + 20 , align ) ;
2014-10-02 10:02:33 +00:00
//doredraw = true;
2014-04-23 17:20:38 +00:00
} else if ( event - > buttons ( ) & Qt : : RightButton ) { // Right Mouse button dragging
m_graphview - > setPointClicked ( event - > pos ( ) ) ;
x - = left ;
x2 - = left ;
if ( ! m_blockzoom ) {
xx = max_x - min_x ;
xmult = xx / double ( w ) ;
qint64 j1 = xmult * x ;
qint64 j2 = xmult * x2 ;
qint64 jj = j2 - j1 ;
min_x + = jj ;
max_x + = jj ;
if ( min_x < rmin_x ) {
min_x = rmin_x ;
max_x = rmin_x + xx ;
}
if ( max_x > rmax_x ) {
max_x = rmax_x ;
min_x = rmax_x - xx ;
}
m_graphview - > SetXBounds ( min_x , max_x , m_group , false ) ;
2014-10-02 10:02:33 +00:00
// doredraw = true;
2014-04-23 17:20:38 +00:00
} else {
qint64 qq = rmax_x - rmin_x ;
xx = max_x - min_x ;
if ( xx = = qq ) { xx = 1800000 ; }
xmult = qq / double ( w ) ;
qint64 j1 = ( xmult * x ) ;
min_x = rmin_x + j1 - ( xx / 2 ) ;
max_x = min_x + xx ;
if ( min_x < rmin_x ) {
min_x = rmin_x ;
max_x = rmin_x + xx ;
}
if ( max_x > rmax_x ) {
max_x = rmax_x ;
min_x = rmax_x - xx ;
}
m_graphview - > SetXBounds ( min_x , max_x , m_group , false ) ;
2014-10-02 10:02:33 +00:00
//doredraw = true;
2014-04-23 17:20:38 +00:00
}
}
}
//if (!nolayer) { // no mouse button
2014-08-29 04:02:16 +00:00
// if (doredraw) {
// m_graphview->timedRedraw(0);
// }
2014-04-23 17:20:38 +00:00
//}
//if (x>left+m_marginleft && x<m_lastbounds.width()-(right+m_marginright) && y>top+m_margintop && y<m_lastbounds.height()-(bottom+m_marginbottom)) { // main area
// x-=left+m_marginleft;
// y-=top+m_margintop;
// //qDebug() << m_title << "Moved" << x << y << left << right << top << bottom << m_width << m_height;
// }
}
2014-08-11 01:20:09 +00:00
bool gGraph : : selectingArea ( ) { return m_selecting_area | | m_graphview - > metaSelect ( ) ; }
2014-04-23 17:20:38 +00:00
void gGraph : : mousePressEvent ( QMouseEvent * event )
{
int y = event - > pos ( ) . y ( ) ;
int x = event - > pos ( ) . x ( ) ;
2020-07-07 18:07:44 +00:00
// qDebug() << m_title << "gGraph mousePressEvent, x=" << x << " y=" << y;
2018-05-05 21:58:11 +00:00
for ( const auto & layer : m_layers ) {
if ( layer - > m_rect . contains ( x , y ) )
if ( layer - > mousePressEvent ( event , this ) ) {
2014-04-23 17:20:38 +00:00
return ;
}
}
/*
int w = m_lastbounds . width ( ) - ( right + m_marginright ) ;
//int h=m_lastbounds.height()-(bottom+m_marginbottom);
//int x2,y2;
double xx = max_x - min_x ;
//double xmult=xx/w;
if ( x > left + m_marginleft & & x < m_lastbounds . width ( ) - ( right + m_marginright ) & & y > top + m_margintop & & y < m_lastbounds . height ( ) - ( bottom + m_marginbottom ) ) { // main area
x - = left + m_marginleft ;
y - = top + m_margintop ;
} */
//qDebug() << m_title << "Clicked" << x << y << left << right << top << bottom << m_width << m_height;
}
void gGraph : : mouseReleaseEvent ( QMouseEvent * event )
{
2014-07-25 07:53:48 +00:00
2014-04-23 17:20:38 +00:00
int y = event - > pos ( ) . y ( ) ;
int x = event - > pos ( ) . x ( ) ;
2020-07-07 18:07:44 +00:00
// qDebug() << m_title << "gGraph mouseReleaseEvent at x,y" << x << y;
2014-04-23 17:20:38 +00:00
2018-05-05 21:58:11 +00:00
for ( const auto & layer : m_layers ) {
if ( layer - > m_rect . contains ( x , y ) )
if ( layer - > mouseReleaseEvent ( event , this ) ) {
2014-04-23 17:20:38 +00:00
return ;
}
}
x - = m_rect . left ( ) ;
y - = m_rect . top ( ) ;
int w = m_rect . width ( ) - left - right ; //(m_marginleft+left+right+m_marginright);
int h = m_rect . height ( ) - bottom ; //+m_marginbottom);
int x2 = m_graphview - > pointClicked ( ) . x ( ) - m_rect . left ( ) ;
2018-03-28 06:22:42 +00:00
//int y2 = m_graphview->pointClicked().y() - m_rect.top();
2014-04-23 17:20:38 +00:00
2014-07-20 09:26:59 +00:00
m_selDurString = QString ( ) ;
2020-07-07 18:07:44 +00:00
// qDebug() << m_title << "Released" << min_x << max_x << x << y << x2 << left << right << top << bottom << m_width << m_height;
2014-04-23 17:20:38 +00:00
if ( m_selecting_area ) {
m_selecting_area = false ;
m_selection . setWidth ( 0 ) ;
2020-07-27 16:57:16 +00:00
// Shift-drag only measures a range and should not zoom in.
if ( ( event - > modifiers ( ) & Qt : : ShiftModifier ) ! = 0 ) {
return ;
}
2014-04-23 17:20:38 +00:00
if ( m_graphview - > horizTravel ( ) > mouse_movement_threshold ) {
x - = left ; //+m_marginleft;
2018-03-28 06:22:42 +00:00
//y -= top; //+m_margintop;
2014-04-23 17:20:38 +00:00
x2 - = left ; //+m_marginleft;
2018-03-28 06:22:42 +00:00
//y2 -= top; //+m_margintop;
2014-04-23 17:20:38 +00:00
if ( x < 0 ) { x = 0 ; }
if ( x2 < 0 ) { x2 = 0 ; }
if ( x > w ) { x = w ; }
if ( x2 > w ) { x2 = w ; }
double xx ;
double xmult ;
if ( ! m_blockzoom ) {
xx = max_x - min_x ;
xmult = xx / double ( w ) ;
qint64 j1 = min_x + xmult * x ;
qint64 j2 = min_x + xmult * x2 ;
qint64 a1 = MIN ( j1 , j2 )
qint64 a2 = MAX ( j1 , j2 )
//if (a1<rmin_x) a1=rmin_x;
if ( a2 > rmax_x ) { a2 = rmax_x ; }
2020-07-07 18:07:44 +00:00
if ( a1 = = a2 ) // Don't zoom into a block if range is zero
return ;
2014-04-23 17:20:38 +00:00
if ( a1 < = rmin_x & & a2 < = rmin_x ) {
//qDebug() << "Foo??";
} else {
2020-07-07 18:07:44 +00:00
// qDebug() << m_title << "!m_blockzoom (960), a1, a2" << a1 << a2;
2014-04-23 17:20:38 +00:00
if ( a2 - a1 < zoom_hard_limit ) { a2 = a1 + zoom_hard_limit ; }
m_graphview - > SetXBounds ( a1 , a2 , m_group ) ;
}
} else {
xx = rmax_x - rmin_x ;
xmult = xx / double ( w ) ;
qint64 j1 = rmin_x + xmult * x ;
qint64 j2 = rmin_x + xmult * x2 ;
qint64 a1 = MIN ( j1 , j2 )
2014-07-19 14:34:32 +00:00
qint64 a2 = MAX ( j1 , j2 )
2014-04-23 17:20:38 +00:00
2014-07-19 14:34:32 +00:00
//if (a1<rmin_x) a1=rmin_x;
2014-04-23 17:20:38 +00:00
if ( a2 > rmax_x ) { a2 = rmax_x ; }
if ( a1 < = rmin_x & & a2 < = rmin_x ) {
qDebug ( ) < < " Foo2?? " ;
} else {
2020-07-07 18:07:44 +00:00
// qDebug() << m_title << "m_blockzoom (979), a1, a2" << a1 << a2;
2014-04-23 17:20:38 +00:00
if ( a2 - a1 < zoom_hard_limit ) { a2 = a1 + zoom_hard_limit ; }
m_graphview - > SetXBounds ( a1 , a2 , m_group ) ;
}
}
return ;
} else { m_graphview - > redraw ( ) ; }
}
if ( ( m_graphview - > horizTravel ( ) < mouse_movement_threshold ) & & ( x > left & & x < w + left
& & y > top & & y < h ) ) {
// normal click in main area
if ( ! m_blockzoom ) {
double zoom ;
if ( event - > button ( ) & Qt : : RightButton ) {
zoom = 1.33 ;
if ( event - > modifiers ( ) & Qt : : ControlModifier ) { zoom * = 1.5 ; }
ZoomX ( zoom , x ) ; // Zoom out
return ;
} else if ( event - > button ( ) & Qt : : LeftButton ) {
zoom = 0.75 ;
if ( event - > modifiers ( ) & Qt : : ControlModifier ) { zoom / = 1.5 ; }
ZoomX ( zoom , x ) ; // zoom in.
return ;
}
} else {
x - = left ;
2018-03-28 06:22:42 +00:00
//y -= top;
2014-04-23 17:20:38 +00:00
//w-=m_marginleft+left;
double qq = rmax_x - rmin_x ;
double xmult ;
double xx = max_x - min_x ;
//if (xx==qq) xx=1800000;
xmult = qq / double ( w ) ;
if ( ( xx = = qq ) | | ( x = = m_lastx23 ) ) {
double zoom = 1 ;
if ( event - > button ( ) & Qt : : RightButton ) {
zoom = 1.33 ;
if ( event - > modifiers ( ) & Qt : : ControlModifier ) { zoom * = 1.5 ; }
} else if ( event - > button ( ) & Qt : : LeftButton ) {
zoom = 0.75 ;
if ( event - > modifiers ( ) & Qt : : ControlModifier ) { zoom / = 1.5 ; }
}
xx * = zoom ;
if ( xx < qq / zoom_hard_limit ) { xx = qq / zoom_hard_limit ; }
if ( xx > qq ) { xx = qq ; }
}
double j1 = xmult * x ;
min_x = rmin_x + j1 - ( xx / 2.0 ) ;
max_x = min_x + xx ;
if ( min_x < rmin_x ) {
min_x = rmin_x ;
max_x = rmin_x + xx ;
} else if ( max_x > rmax_x ) {
max_x = rmax_x ;
min_x = rmax_x - xx ;
}
m_graphview - > SetXBounds ( min_x , max_x , m_group ) ;
m_lastx23 = x ;
}
}
//m_graphview->redraw();
}
void gGraph : : wheelEvent ( QWheelEvent * event )
{
2020-07-07 18:07:44 +00:00
qDebug ( ) < < m_title < < " Wheel " < < event - > x ( ) < < event - > y ( ) < < event - > delta ( ) ;
2014-04-23 17:20:38 +00:00
//int y=event->pos().y();
if ( event - > orientation ( ) = = Qt : : Horizontal ) {
return ;
}
int x = event - > pos ( ) . x ( ) - m_graphview - > titleWidth ; //(left+m_marginleft);
if ( event - > delta ( ) > 0 ) {
ZoomX ( 0.75 , x ) ;
} else {
ZoomX ( 1.5 , x ) ;
}
int y = event - > pos ( ) . y ( ) ;
x = event - > pos ( ) . x ( ) ;
2018-05-05 21:58:11 +00:00
for ( const auto & layer : m_layers ) {
if ( layer - > m_rect . contains ( x , y ) ) {
layer - > wheelEvent ( event , this ) ;
2014-04-23 17:20:38 +00:00
}
}
}
void gGraph : : mouseDoubleClickEvent ( QMouseEvent * event )
{
//mousePressEvent(event);
//mouseReleaseEvent(event);
int y = event - > pos ( ) . y ( ) ;
int x = event - > pos ( ) . x ( ) ;
2018-05-05 21:58:11 +00:00
for ( const auto & layer : m_layers ) {
if ( layer - > m_rect . contains ( x , y ) ) {
layer - > mouseDoubleClickEvent ( event , this ) ;
2014-04-23 17:20:38 +00:00
}
}
//int w=m_lastbounds.width()-(m_marginleft+left+right+m_marginright);
//int h=m_lastbounds.height()-(bottom+m_marginbottom);
//int x2=m_graphview->pointClicked().x(),y2=m_graphview->pointClicked().y();
// if ((m_graphview->horizTravel()<mouse_movement_threshold) && (x>left+m_marginleft && x<w+m_marginleft+left && y>top+m_margintop && y<h)) { // normal click in main area
// if (event->button() & Qt::RightButton) {
// ZoomX(1.66,x); // Zoon out
// return;
// } else if (event->button() & Qt::LeftButton) {
// ZoomX(0.75/2.0,x); // zoom in.
// return;
// }
// } else {
// Propagate the events to graph Layers
// }
//mousePressEvent(event);
//mouseReleaseEvent(event);
//qDebug() << m_title << "Double Clicked" << event->x() << event->y();
}
void gGraph : : keyPressEvent ( QKeyEvent * event )
{
2018-05-05 21:58:11 +00:00
for ( const auto & layer : m_layers ) {
layer - > keyPressEvent ( event , this ) ;
2014-04-23 17:20:38 +00:00
}
//qDebug() << m_title << "Key Pressed.. implement me" << event->key();
}
2014-07-19 12:17:32 +00:00
void gGraph : : keyReleaseEvent ( QKeyEvent * event )
{
if ( ! m_graphview ) return ;
if ( m_graphview - > selectionInProgress ( ) & & m_graphview - > metaSelect ( ) ) {
2014-07-20 09:22:31 +00:00
if ( ! ( event - > modifiers ( ) & Qt : : AltModifier ) ) {
2014-07-19 12:17:32 +00:00
}
}
}
2014-04-23 17:20:38 +00:00
void gGraph : : ZoomX ( double mult , int origin_px )
{
2020-07-07 18:07:44 +00:00
// qDebug() << "gGraph::ZoomX width, left, right" << m_rect.width() << left << right;
2014-04-23 17:20:38 +00:00
int width = m_rect . width ( ) - left - right ; //(m_marginleft+left+right+m_marginright);
if ( origin_px = = 0 ) { origin_px = ( width / 2 ) ; }
else { origin_px - = left ; }
if ( origin_px < 0 ) { origin_px = 0 ; }
if ( origin_px > width ) { origin_px = width ; }
// Okay, I want it to zoom in centered on the mouse click area..
// Find X graph position of mouse click
// find current zoom width
// apply zoom
// center on point found in step 1.
qint64 min = min_x ;
qint64 max = max_x ;
double hardspan = rmax_x - rmin_x ;
double span = max - min ;
double ww = double ( origin_px ) / double ( width ) ;
double origin = ww * span ;
//double center=0.5*span;
//double dist=(origin-center);
double q = span * mult ;
if ( q > hardspan ) { q = hardspan ; }
if ( q < hardspan / zoom_hard_limit ) { q = hardspan / zoom_hard_limit ; }
min = min + origin - ( q * ww ) ;
max = min + q ;
if ( min < rmin_x ) {
min = rmin_x ;
max = min + q ;
}
if ( max > rmax_x ) {
max = rmax_x ;
min = max - q ;
}
2014-10-02 10:02:33 +00:00
//extern const int max_history;
2014-07-19 14:09:04 +00:00
2014-04-23 17:20:38 +00:00
m_graphview - > SetXBounds ( min , max , m_group ) ;
//updateSelectionTime(max-min);
}
2014-05-10 00:50:35 +00:00
void gGraph : : DrawTextQue ( QPainter & painter )
2014-04-23 17:20:38 +00:00
{
2018-06-10 04:57:14 +00:00
AppSetting - > usePixmapCaching ( ) ? m_graphview - > DrawTextQueCached ( painter ) : m_graphview - > DrawTextQue ( painter ) ;
2014-04-23 17:20:38 +00:00
}
// margin recalcs..
void gGraph : : resize ( int width , int height )
{
invalidate_xAxisImage = true ;
invalidate_yAxisImage = true ;
Q_UNUSED ( width ) ;
Q_UNUSED ( height ) ;
//m_height=height;
//m_width=width;
}
qint64 gGraph : : MinX ( )
{
qint64 val = 0 , tmp ;
2018-05-05 21:58:11 +00:00
for ( const auto & layer : m_layers ) {
if ( layer - > isEmpty ( ) ) {
2014-04-23 17:20:38 +00:00
continue ;
}
2018-05-05 21:58:11 +00:00
tmp = layer - > Minx ( ) ;
2014-04-23 17:20:38 +00:00
if ( ! tmp ) {
continue ;
}
if ( ! val | | tmp < val ) {
val = tmp ;
}
}
if ( val ) { rmin_x = val ; }
return val ;
}
qint64 gGraph : : MaxX ( )
{
//bool first=true;
qint64 val = 0 , tmp ;
2018-05-05 21:58:11 +00:00
for ( const auto & layer : m_layers ) {
if ( layer - > isEmpty ( ) ) {
2014-04-23 17:20:38 +00:00
continue ;
}
2018-05-05 21:58:11 +00:00
tmp = layer - > Maxx ( ) ;
2014-04-23 17:20:38 +00:00
//if (!tmp) continue;
2018-05-05 21:58:11 +00:00
if ( ! val | | ( tmp > val ) ) {
2014-04-23 17:20:38 +00:00
val = tmp ;
}
}
if ( val ) { rmax_x = val ; }
return val ;
}
EventDataType gGraph : : MinY ( )
{
bool first = true ;
EventDataType val = 0 , tmp ;
if ( m_enforceMinY ) {
return rmin_y = f_miny ;
}
2018-05-05 21:58:11 +00:00
for ( const auto & layer : m_layers ) {
if ( layer - > isEmpty ( ) | | ( layer - > layerType ( ) = = LT_Other ) ) {
2014-04-23 17:20:38 +00:00
continue ;
}
2018-05-05 21:58:11 +00:00
tmp = layer - > Miny ( ) ;
2014-04-23 17:20:38 +00:00
2014-08-27 09:00:55 +00:00
// if (tmp == 0 && tmp == (*l)->Maxy()) {
// continue;
// }
2014-04-23 17:20:38 +00:00
if ( first ) {
val = tmp ;
first = false ;
} else {
if ( tmp < val ) {
val = tmp ;
}
}
}
return rmin_y = val ;
2021-09-19 04:02:51 +00:00
// return rmin_y = val * 0.9;
2014-04-23 17:20:38 +00:00
}
EventDataType gGraph : : MaxY ( )
{
bool first = true ;
EventDataType val = 0 , tmp ;
if ( m_enforceMaxY ) {
return rmax_y = f_maxy ;
}
2018-05-05 21:58:11 +00:00
for ( const auto & layer : m_layers ) {
2014-08-27 04:36:40 +00:00
if ( layer - > isEmpty ( ) | | ( layer - > layerType ( ) = = LT_Other ) ) {
2014-04-23 17:20:38 +00:00
continue ;
}
tmp = layer - > Maxy ( ) ;
2014-08-27 09:00:55 +00:00
// if (tmp == 0 && layer->Miny() == 0) {
// continue;
// }
2014-04-23 17:20:38 +00:00
if ( first ) {
val = tmp ;
first = false ;
} else {
if ( tmp > val ) {
val = tmp ;
}
}
}
return rmax_y = val ;
}
EventDataType gGraph : : physMinY ( )
{
bool first = true ;
EventDataType val = 0 , tmp ;
//if (m_enforceMinY) return rmin_y=f_miny;
2018-05-05 21:58:11 +00:00
for ( const auto & layer : m_layers ) {
2014-04-23 17:20:38 +00:00
if ( layer - > isEmpty ( ) ) {
continue ;
}
tmp = layer - > physMiny ( ) ;
if ( tmp = = 0 & & layer - > physMaxy ( ) = = 0 ) {
continue ;
}
if ( first ) {
val = tmp ;
first = false ;
} else {
if ( tmp < val ) {
val = tmp ;
}
}
}
return rphysmin_y = val ;
}
EventDataType gGraph : : physMaxY ( )
{
bool first = true ;
EventDataType val = 0 , tmp ;
// if (m_enforceMaxY) return rmax_y=f_maxy;
2018-05-05 21:58:11 +00:00
for ( const auto & layer : m_layers ) {
2014-04-23 17:20:38 +00:00
if ( layer - > isEmpty ( ) ) {
continue ;
}
tmp = layer - > physMaxy ( ) ;
if ( tmp = = 0 & & layer - > physMiny ( ) = = 0 ) {
continue ;
}
if ( first ) {
val = tmp ;
first = false ;
} else {
if ( tmp > val ) {
val = tmp ;
}
}
}
return rphysmax_y = val ;
}
void gGraph : : SetMinX ( qint64 v )
{
rmin_x = min_x = v ;
}
void gGraph : : SetMaxX ( qint64 v )
{
rmax_x = max_x = v ;
}
void gGraph : : SetMinY ( EventDataType v )
{
rmin_y = min_y = v ;
}
void gGraph : : SetMaxY ( EventDataType v )
{
rmax_y = max_y = v ;
}
Layer * gGraph : : getLineChart ( )
{
gLineChart * lc ;
2018-05-05 21:58:11 +00:00
for ( auto & layer : m_layers ) {
lc = dynamic_cast < gLineChart * > ( layer ) ;
2014-04-23 17:20:38 +00:00
if ( lc ) { return lc ; }
}
return nullptr ;
}
2014-08-09 17:12:37 +00:00
int gGraph : : minHeight ( )
{
int minheight = m_min_height ;
2014-10-02 07:56:57 +00:00
// int top = 0;
// int center = 0;
// int bottom = 0;
2018-05-05 21:58:11 +00:00
for ( const auto & layer : m_layers ) {
int mh = layer - > minimumHeight ( ) ;
2014-08-09 17:12:37 +00:00
mh + = m_margintop + m_marginbottom ;
if ( mh > minheight ) minheight = mh ;
}
// layers need to set their own too..
return minheight ;
}
2014-04-23 17:20:38 +00:00
int GetXHeight ( QFont * font )
{
QFontMetrics fm ( * font ) ;
return fm . xHeight ( ) ;
}
2014-07-21 16:14:07 +00:00
void gGraph : : dumpInfo ( ) {
2018-05-05 21:58:11 +00:00
for ( const auto & layer : m_layers ) {
if ( ! layer - > visible ( ) ) { continue ; }
2014-07-21 16:14:07 +00:00
2018-05-05 21:58:11 +00:00
if ( layer - > position ( ) = = LayerCenter ) {
gLineChart * lc = dynamic_cast < gLineChart * > ( layer ) ;
2014-07-21 16:14:07 +00:00
if ( lc ! = nullptr ) {
QString text = lc - > getMetaString ( currentTime ( ) ) ;
if ( ! text . isEmpty ( ) ) {
mainwin - > log ( text ) ;
}
}
}
}
}
2014-08-27 13:42:10 +00:00