diff --git a/SleepyHeadQT.pro b/SleepyHeadQT.pro index ce1fd44f..970e7f6a 100644 --- a/SleepyHeadQT.pro +++ b/SleepyHeadQT.pro @@ -5,11 +5,12 @@ SUBDIRS += sleepyhead CONFIG += ordered TRANSLATIONS += \ - Translations/Nederlands.nl_NL.ts \ + Translations/Nederlands.nl.ts \ Translations/Francais.fr.ts \ Translations/Svenska.se.ts \ Translations/Deutsch.de_DE.ts \ Translations/Espaniol.es.ts \ - Translations/Bulgarian.bg.ts - + Translations/Bulgarian.bg.ts \ + Translations/Chinese.ch.ts \ + Translations/Chinese_Simplified.ch_s.ts diff --git a/sleepyhead/Graphs/gGraphView.cpp b/sleepyhead/Graphs/gGraphView.cpp index 96f55681..dc2ce6fb 100644 --- a/sleepyhead/Graphs/gGraphView.cpp +++ b/sleepyhead/Graphs/gGraphView.cpp @@ -1145,6 +1145,7 @@ gGraph::gGraph(gGraphView *graphview,QString title,QString units, int height,sho m_marginleft=0; m_marginright=15; m_selecting_area=m_blockzoom=false; + m_pinned=false; m_lastx23=0; invalidate_yAxisImage=true; @@ -3108,9 +3109,32 @@ bool gGraphView::renderGraphs() } else threaded=false; */ //#endif //threaded=false; + + + lines_drawn_this_frame=0; + quads_drawn_this_frame=0; + + // Calculate the height of pinned graphs + + float pinned_height=0; // pixel height total + int pinned_graphs=0; // count + for (int i=0;iisEmpty()) continue; if (!m_graphs[i]->visible()) continue; + if (!m_graphs[i]->isPinned()) continue; + h=m_graphs[i]->height() * m_scaleY; + pinned_height+=h+graphSpacer; + pinned_graphs++; + } + + py+=pinned_height; // start drawing at the end of pinned space + + // Draw non pinned graphs + for (int i=0;iisEmpty()) continue; + if (!m_graphs[i]->visible()) continue; + if (m_graphs[i]->isPinned()) continue; numgraphs++; h=m_graphs[i]->height() * m_scaleY; @@ -3139,6 +3163,66 @@ bool gGraphView::renderGraphs() py=ceil(py+h+graphSpacer); } + // Physically draw the unpinned graphs + int s=m_drawlist.size(); + for (int i=0;ipaint(g->m_rect.x(), g->m_rect.y(), g->m_rect.width(), g->m_rect.height()); + } + backlines->draw(); + for (int i=0;idrawGLBuf(); + } + quads->draw(); + lines->draw(); + DrawTextQue(); + + py=0; // start drawing at top... + + // glEnable(GL_BLEND); + glBlendFunc(GL_SRC_ALPHA, GL_ONE_MINUS_SRC_ALPHA); + glBegin(GL_QUADS); + glColor4f(1.0,1.0,1.0,1.0); // Gradient start + glVertex2f(0, pinned_height); + glVertex2f(0, 0); + glColor4f(0.7,0.7,1.0,1.0); // Gradient End + glVertex2f(width(), 0); + glVertex2f(width(), pinned_height); + glEnd(); + // glDisable(GL_BLEND); + + + // Draw Pinned graphs + for (int i=0;iisEmpty()) continue; + if (!m_graphs[i]->visible()) continue; + if (!m_graphs[i]->isPinned()) continue; + h=m_graphs[i]->height() * m_scaleY; + numgraphs++; + if (py > height()) + break; // we are done.. can't draw anymore + + if ((py + h + graphSpacer) >= 0) { + w=width(); + int tw=(m_graphs[i]->showTitle() ? titleWidth : 0); + + queGraph(m_graphs[i],px+tw,py,width()-tw,h); + + if (m_showsplitter) { + // draw the splitter handle + QColor ca=QColor(128,128,128,255); + backlines->add(0, py+h, w, py+h, ca.rgba()); + ca=QColor(192,192,192,255); + backlines->add(0, py+h+1, w, py+h+1, ca.rgba()); + ca=QColor(90,90,90,255); + backlines->add(0, py+h+2, w, py+h+2, ca.rgba()); + } + + } + py=ceil(py+h+graphSpacer); + } + //int thr=m_idealthreads; #ifdef ENABLED_THREADED_DRAWING for (int i=0;idraw(); for (int i=0;idraw(); lines->draw(); -// lines->setSize(linesize); + + + // lines->setSize(linesize); // DrawTextQue(); //glDisable(GL_TEXTURE_2D); @@ -3480,6 +3564,11 @@ void gGraphView::mouseMoveEvent(QMouseEvent * event) if (y < yy) { for (int i=m_graph_index-1;i>=0;i--) { + if (m_graphs[i]->isPinned()!=m_graphs[m_graph_index]->isPinned()) { + m_graph_dragging=false; + // fix cursor + break; + } empty=m_graphs[i]->isEmpty() || (!m_graphs[i]->visible()); // swapping upwards. int yy2=yy-graphSpacer-m_graphs[i]->height()*m_scaleY; @@ -3502,6 +3591,11 @@ void gGraphView::mouseMoveEvent(QMouseEvent * event) // swapping downwards //qDebug() << "Graph Reorder" << m_graph_index; for (int i=m_graph_index+1;iisPinned()!=m_graphs[m_graph_index]->isPinned()) { + m_graph_dragging=false; + // fix cursor + break; + } empty=m_graphs[i]->isEmpty() || (!m_graphs[i]->visible()); p=m_graphs[m_graph_index]; m_graphs[m_graph_index]=m_graphs[i]; @@ -3517,13 +3611,60 @@ void gGraphView::mouseMoveEvent(QMouseEvent * event) return; } - float py = -m_offsetY; - float h; + float py = 0, pinned_height=0, h; + bool done=false; - // Propagate mouseMove events to relevant graphs + // Do pinned graphs first for (int i=0; i < m_graphs.size(); i++) { - if (m_graphs[i]->isEmpty() || (!m_graphs[i]->visible())) + if (m_graphs[i]->isEmpty() || !m_graphs[i]->visible() || !m_graphs[i]->isPinned()) + continue; + + h=m_graphs[i]->height() * m_scaleY; + pinned_height += h + graphSpacer; + + if (py > height()) + break; // we are done.. can't draw anymore + + if (!((y >= py+m_graphs[i]->top) && (y < py + h-m_graphs[i]->bottom))) { + if (m_graphs[i]->isSelected()) { + m_graphs[i]->deselect(); + timedRedraw(150); + } + } + + // Update Mouse Cursor shape + if ((y >= py + h -1) && (y < (py + h + graphSpacer))) { + this->setCursor(Qt::SplitVCursor); + done=true; + } else if ((y >= py+1) && (y < py + h)) { + if (x >= titleWidth+10) + this->setCursor(Qt::ArrowCursor); + else + this->setCursor(Qt::OpenHandCursor); + + + m_horiz_travel+=qAbs(x-m_lastxpos)+qAbs(y-m_lastypos); + m_lastxpos=x; + m_lastypos=y; +// QPoint p(x,y); +// QMouseEvent e(event->type(),p,event->button(),event->buttons(),event->modifiers()); + m_graphs[i]->mouseMoveEvent(event); + + done=true; + } + py += h + graphSpacer; + + } + + py = -m_offsetY; + py += pinned_height; + + // Propagate mouseMove events to relevant graphs + if (!done) + for (int i=0; i < m_graphs.size(); i++) { + + if (m_graphs[i]->isEmpty() || !m_graphs[i]->visible() || m_graphs[i]->isPinned()) continue; h=m_graphs[i]->height() * m_scaleY; @@ -3612,8 +3753,7 @@ void gGraphView::mouseMoveEvent(QMouseEvent * event) } */ // } - py+=h; - py+=graphSpacer; + py += h + graphSpacer; } } @@ -3623,14 +3763,76 @@ void gGraphView::mousePressEvent(QMouseEvent * event) int x=event->x(); int y=event->y(); - float py=-m_offsetY; - float h; + float h,pinned_height=0,py=0; + bool done=false; + + // first handle pinned graphs. + // Calculate total height of all pinned graphs + for (int i=0;iisEmpty() + || !m_graphs[i]->visible() + || !m_graphs[i]->isPinned()) + continue; + + h=m_graphs[i]->height() * m_scaleY; + pinned_height += h+graphSpacer; + + if (py>height()) + break; + + if ((py + h + graphSpacer) >= 0) { + if ((y >= py + h-1) && (y <= py + h + graphSpacer)) { + this->setCursor(Qt::SplitVCursor); + m_sizer_dragging=true; + m_sizer_index=i; + m_sizer_point.setX(x); + m_sizer_point.setY(y); + done=true; + } else if ((y >= py) && (y < py + h)) { + //qDebug() << "Clicked" << i; + if (x < titleWidth+20) { + // clicked on title to drag graph.. + // Note: reorder has to be limited to pinned graphs. + m_graph_dragging=true; + m_tooltip->cancel(); + + timedRedraw(50); + m_graph_index=i; + m_sizer_point.setX(x); + m_sizer_point.setY(py); // point at top of graph.. + this->setCursor(Qt::ClosedHandCursor); + } + + { // send event to graph.. + m_point_clicked=QPoint(event->x(),event->y()); + //QMouseEvent e(event->type(),m_point_clicked,event->button(),event->buttons(),event->modifiers()); + m_button_down=true; + m_horiz_travel=0; + m_graph_index=i; + m_selected_graph=m_graphs[i]; + m_graphs[i]->mousePressEvent(event); + } + done=true; + } + + } + py += h + graphSpacer; + } + + + + // then handle the remainder... + py=-m_offsetY; + py+=pinned_height; + + if (!done) for (int i=0;iisEmpty() || (!m_graphs[i]->visible())) continue; + if (m_graphs[i]->isEmpty() || !m_graphs[i]->visible() || m_graphs[i]->isPinned()) continue; h=m_graphs[i]->height()*m_scaleY; + if (py>height()) break; @@ -3666,9 +3868,7 @@ void gGraphView::mousePressEvent(QMouseEvent * event) } } - py+=h; - py+=graphSpacer; - + py += h + graphSpacer; } } @@ -3677,15 +3877,51 @@ void gGraphView::mouseReleaseEvent(QMouseEvent * event) int x=event->x(); int y=event->y(); - float py = -m_offsetY; - float h; + float h,py=0,pinned_height=0; + bool done=false; + + // Handle pinned graphs first for (int i=0; i < m_graphs.size(); i++) { - - if (m_graphs[i]->isEmpty() || (!m_graphs[i]->visible())) + if (m_graphs[i]->isEmpty() || !m_graphs[i]->visible() || !m_graphs[i]->isPinned()) continue; h=m_graphs[i]->height() * m_scaleY; + pinned_height += h+graphSpacer; + + if (py > height()) + break; // we are done.. can't draw anymore + + if ((y >= py + h -1) && (y < (py + h + graphSpacer))) { + this->setCursor(Qt::SplitVCursor); + done=true; + } else if ((y >= py+1) && (y <= py + h)) { + +// if (!m_sizer_dragging && !m_graph_dragging) { +// m_graphs[i]->mouseReleaseEvent(event); +// } + + if (x >= titleWidth+10) + this->setCursor(Qt::ArrowCursor); + else + this->setCursor(Qt::OpenHandCursor); + done=true; + } + py += h + graphSpacer; + } + + // Now do the unpinned ones + py = -m_offsetY; + py += pinned_height; + + if (done) + for (int i=0; i < m_graphs.size(); i++) { + + if (m_graphs[i]->isEmpty() || !m_graphs[i]->visible() || m_graphs[i]->isPinned()) + continue; + + h=m_graphs[i]->height() * m_scaleY; + if (py > height()) break; // we are done.. can't draw anymore @@ -3702,7 +3938,7 @@ void gGraphView::mouseReleaseEvent(QMouseEvent * event) else this->setCursor(Qt::OpenHandCursor); } - + py += h + graphSpacer; } if (m_sizer_dragging) { @@ -3728,17 +3964,53 @@ void gGraphView::mouseReleaseEvent(QMouseEvent * event) void gGraphView::mouseDoubleClickEvent(QMouseEvent * event) { - mousePressEvent(event); + mousePressEvent(event); // signal missing.. a qt change might "fix" this if we are not careful. int x=event->x(); int y=event->y(); - float py=-m_offsetY; - float h; + float h,py=0,pinned_height=0; + bool done=false; + // Handle pinned graphs first + for (int i=0; i < m_graphs.size(); i++) { + if (m_graphs[i]->isEmpty() || !m_graphs[i]->visible() || !m_graphs[i]->isPinned()) + continue; + + h=m_graphs[i]->height() * m_scaleY; + pinned_height += h + graphSpacer; + + if (py > height()) + break; // we are done.. can't draw anymore + + if ((py + h + graphSpacer) >= 0) { + if ((y >= py) && (y <= py + h)) { + if (x < titleWidth) { + // What to do when double clicked on the graph title ?? + m_graphs[i]->mouseDoubleClickEvent(event); + } else { + // send event to graph.. + m_graphs[i]->mouseDoubleClickEvent(event); + } + done=true; + } else if ((y >= py + h) && (y <= py + h + graphSpacer + 1)) { + // What to do when double clicked on the resize handle? + done=true; + } + } + py+=h; + py+=graphSpacer; // do we want the extra spacer down the bottom? + } + + + py=-m_offsetY; + py+=pinned_height; + + if (!done) // then handle unpinned graphs for (int i=0;iisEmpty()) continue; + if (m_graphs[i]->isEmpty() || !m_graphs[i]->visible() || m_graphs[i]->isPinned()) + continue; h=m_graphs[i]->height()*m_scaleY; if (py>height()) @@ -4002,7 +4274,7 @@ void MyScrollBar::SendWheelEvent(QWheelEvent * e) } const quint32 gvmagic=0x41756728; -const quint16 gvversion=1; +const quint16 gvversion=2; void gGraphView::SaveSettings(QString title) { @@ -4024,6 +4296,7 @@ void gGraphView::SaveSettings(QString title) out << m_graphs[i]->RecMinY(); out << m_graphs[i]->RecMaxY(); out << m_graphs[i]->zoomY(); + out << (bool)m_graphs[i]->isPinned(); } f.close(); @@ -4060,6 +4333,7 @@ bool gGraphView::LoadSettings(QString title) float hght; bool vis; EventDataType recminy,recmaxy; + bool pinned; short zoomy=0; @@ -4075,6 +4349,9 @@ bool gGraphView::LoadSettings(QString title) if (gvversion>=1) { in >> zoomy; } + if (gvversion>=2) { + in >> pinned; + } gi=m_graphsbytitle.find(name); if (gi==m_graphsbytitle.end()) { qDebug() << "Graph" << name << "has been renamed or removed"; @@ -4086,6 +4363,7 @@ bool gGraphView::LoadSettings(QString title) g->setRecMinY(recminy); g->setRecMaxY(recmaxy); g->setZoomY(zoomy); + g->setPinned(pinned); } } diff --git a/sleepyhead/Graphs/gGraphView.h b/sleepyhead/Graphs/gGraphView.h index c03f1d86..7f797b1f 100644 --- a/sleepyhead/Graphs/gGraphView.h +++ b/sleepyhead/Graphs/gGraphView.h @@ -760,6 +760,9 @@ public: const inline QRect & rect() { return m_rect; } + bool isPinned() { return m_pinned; } + void setPinned(bool b) { m_pinned=b; } + // //! \brief Returns the main gGraphView objects gVertexBuffer stippled line list. //GLShortBuffer * stippled(); @@ -831,6 +834,7 @@ protected: bool m_enforceMinY,m_enforceMaxY; bool m_showTitle; bool m_printing; + bool m_pinned; short m_zoomY; diff --git a/sleepyhead/daily.cpp b/sleepyhead/daily.cpp index 0940e71e..74341f22 100644 --- a/sleepyhead/daily.cpp +++ b/sleepyhead/daily.cpp @@ -119,7 +119,9 @@ Daily::Daily(QWidget *parent,gGraphView * shared) int default_height=PROFILE.appearance->graphHeight(); SF=new gGraph(GraphView,STR_TR_EventFlags,STR_TR_EventFlags,default_height); + SF->setPinned(true); FRW=new gGraph(GraphView,STR_TR_FlowRate, schema::channel[CPAP_FlowRate].fullname()+"\n"+schema::channel[CPAP_FlowRate].description()+"\n("+schema::channel[CPAP_FlowRate].units()+")",default_height); + //FRW->setPinned(true); if (PROFILE.general->calculateRDI()) { @@ -360,6 +362,10 @@ Daily::Daily(QWidget *parent,gGraphView * shared) ui->evViewLCD->display(ews); GraphView->LoadSettings("Daily"); + + ui->pinFlowButton->setChecked(FRW->isPinned()); + ui->pinFlagsButton->setChecked(SF->isPinned()); + icon_on=new QIcon(":/icons/session-on.png"); icon_off=new QIcon(":/icons/session-off.png"); @@ -2154,3 +2160,15 @@ void Daily::on_resetLayoutButton_clicked() { GraphView->resetLayout(); } + +void Daily::on_pinFlagsButton_toggled(bool checked) +{ + SF->setPinned(checked); + GraphView->redraw(); +} + +void Daily::on_pinFlowButton_toggled(bool checked) +{ + FRW->setPinned(checked); + GraphView->redraw(); +} diff --git a/sleepyhead/daily.h b/sleepyhead/daily.h index 6726cf3f..0e1b18f4 100644 --- a/sleepyhead/daily.h +++ b/sleepyhead/daily.h @@ -248,6 +248,11 @@ private slots: void on_weightSpinBox_valueChanged(double arg1); void doToggleSession(Session *); + + void on_pinFlagsButton_toggled(bool checked); + + void on_pinFlowButton_toggled(bool checked); + protected: private: diff --git a/sleepyhead/daily.ui b/sleepyhead/daily.ui index 825a1f3d..199c9e25 100644 --- a/sleepyhead/daily.ui +++ b/sleepyhead/daily.ui @@ -23,7 +23,16 @@ 0 - + + 0 + + + 0 + + + 0 + + 0 @@ -52,7 +61,16 @@ 0 - + + 0 + + + 0 + + + 0 + + 0 @@ -405,7 +423,16 @@ QToolButton:pressed { 0 - + + 0 + + + 0 + + + 0 + + 0 @@ -434,7 +461,16 @@ QToolButton:pressed { - + + 4 + + + 4 + + + 4 + + 4 @@ -491,7 +527,16 @@ QToolButton:pressed { 0 - + + 0 + + + 0 + + + 0 + + 0 @@ -845,7 +890,16 @@ QSlider::handle:horizontal { Bookmarks - + + 4 + + + 4 + + + 4 + + 4 @@ -998,6 +1052,26 @@ QToolButton:pressed { + + + + Pin Flags + + + true + + + + + + + Pin Flow + + + true + + +