/*************************************************************** * Name: SleepyHeadMain.cpp * Purpose: Code for Application Frame * Author: Mark Watkins (jedimark64@users.sourceforge.net) * Created: 2011-05-20 * Copyright: Mark Watkins (http://sourceforge.net/projects/sleepyhead/) * License: GPL **************************************************************/ #ifdef WX_PRECOMP #include "wx_pch.h" #endif #ifdef __BORLANDC__ #pragma hdrstop #endif //__BORLANDC__ #include "version.h" #include #include #include #include #include #include #include #include #include #include #include #include #include "SleepyHeadMain.h" #include "sleeplib/profiles.h" #include "sleeplib/machine_loader.h" #if defined(__WXMSW__) extern "C" void *_GdipStringFormatCachedGenericTypographic = NULL; #endif wxProgressDialog *loader_progress; //helper functions enum wxbuildinfoformat { short_f, long_f }; wxString wxbuildinfo(wxbuildinfoformat format) { wxString wxbuild(wxVERSION_STRING); if (format == long_f ) { #if defined(__WXMSW__) wxbuild << _T("-Windows"); #elif defined(__WXMAC__) wxbuild << _T("-Mac"); #elif defined(__UNIX__) wxbuild << _T("-Linux"); #endif #if wxUSE_UNICODE wxbuild << _T("-Unicode build"); #else wxbuild << _T("-ANSI build"); #endif // wxUSE_UNICODE } return wxbuild; } SleepyHeadFrame::SleepyHeadFrame(wxFrame *frame) : GUIFrame(frame) { wxString title=wxTheApp->GetAppName()+wxT(" v")+wxString(AutoVersion::FULLVERSION_STRING,wxConvUTF8); SetTitle(title); profile=Profiles::Get(); if (!profile) { wxLogError(wxT("Couldn't get active profile")); abort(); } UpdateProfiles(); if (pref.Exists("ShowSerialNumbers")) ViewMenuSerial->Check(pref["ShowSerialNumbers"]); // wxDisableAsserts(); // Create AUINotebook Tabs wxCommandEvent dummy; OnViewMenuSummary(dummy); // Summary Page OnViewMenuDaily(dummy); // Daily Page this->Connect(wxID_ANY, wxEVT_DO_SCREENSHOT, wxCommandEventHandler(SleepyHeadFrame::DoScreenshot)); //this->Connect(wxID_ANY, wxEVT_MACHINE_SELECTED, wxCommandEventHandler(SleepyHeadFrame::OnMachineSelected)); //this->Connect(wxID_ANY, wxEVT_COMMAND_MENU_SELECTED, wxCommandEventHandler(SleepyHeadFrame::DoScreenshot)); #if wxUSE_STATUSBAR //statusBar->SetStatusText(_("Hello!"), 0); statusBar->SetStatusText(wxbuildinfo(long_f), 1); #endif } SleepyHeadFrame::~SleepyHeadFrame() { //delete summary; //DestroyLoaders(); } void SleepyHeadFrame::UpdateProfiles() { // cpap_machines=profile->GetMachines(MT_CPAP); wxMenuItemList z=ProfileMenu->GetMenuItems(); int i=ProfileMenuID; for (unsigned int j=0;jDisconnect(i,wxEVT_COMMAND_MENU_SELECTED,wxCommandEventHandler(SleepyHeadFrame::OnProfileSelected)); ProfileMenu->Remove(mi); i++; } i=ProfileMenuID; for (map::iterator p=Profiles::profiles.begin();p!=Profiles::profiles.end();p++) { Profile &pro=*(Profiles::profiles[p->first]); wxMenuItem *item=ProfileMenu->AppendRadioItem(i,pro["Realname"],wxEmptyString); if (p->first==pref["Profile"].GetString()) { item->Check(true); } this->Connect(i,wxEVT_COMMAND_MENU_SELECTED,wxCommandEventHandler(SleepyHeadFrame::OnProfileSelected)); i++; } } void SleepyHeadFrame::OnClose(wxCloseEvent &event) { int idx=main_auinotebook->GetPageIndex(daily); if (idx!=wxNOT_FOUND) { daily->Close(); } idx=main_auinotebook->GetPageIndex(summary); if (idx!=wxNOT_FOUND) { summary->Close(); } //delete summary wxMenuItemList z=ProfileMenu->GetMenuItems(); int i=ProfileMenuID; for (unsigned int j=0;jDisconnect(i,wxEVT_COMMAND_MENU_SELECTED,wxCommandEventHandler(SleepyHeadFrame::OnProfileSelected)); ProfileMenu->Remove(mi); i++; } this->Disconnect(wxID_ANY, wxEVT_DO_SCREENSHOT, wxCommandEventHandler(SleepyHeadFrame::DoScreenshot)); Destroy(); } void SleepyHeadFrame::OnQuit(wxCommandEvent &event) { Destroy(); } void SleepyHeadFrame::OnFullscreen(wxCommandEvent& event) { if (!IsFullScreen()) { ShowFullScreen(true,wxFULLSCREEN_NOBORDER|wxFULLSCREEN_NOCAPTION|wxFULLSCREEN_NOSTATUSBAR); } else { ShowFullScreen(false); } } void SleepyHeadFrame::OnProfileSelected(wxCommandEvent& event) { int id=event.GetId()-ProfileMenuID; wxLogMessage(wxT("Profile Selected:")+wxString::Format(wxT("%i"),id)); /*Machine *m=cpap_machines[id]; if (m) { pref[wxT("DefaultMachine")]=(long)id; } int idx=main_auinotebook->GetPageIndex(daily); if (idx!=wxNOT_FOUND) { daily->RefreshData(m); daily->Refresh(); } idx=main_auinotebook->GetPageIndex(summary); if (idx!=wxNOT_FOUND) { summary->ResetProfile(profile) summary->Refresh(); } */ //event.Skip(); //Refresh(); } void SleepyHeadFrame::OnScreenshot(wxCommandEvent& event) { ToolsMenu->UpdateUI(); //wxWindow::DoUpdateWindowUI(); wxWindow::UpdateWindowUI(); //Refresh(true); // Make sure the menu is closed.. (It pushes the Update event in front of the manual event we push next) //Update(true); wxCommandEvent MyEvent( wxEVT_DO_SCREENSHOT); wxPostEvent(this, MyEvent); } void SleepyHeadFrame::DoScreenshot( wxCommandEvent &event ) { wxRect r=GetRect(); #if defined(__UNIX__) // Borrowed.. this need fixing. /*int cx=r.x, cy=r.y; ClientToScreen(&cx,&cy); int border_width = cx - r.x; int title_bar_height = cy - r.y; r.width += (border_width * 2); r.height += title_bar_height + border_width; */ #endif wxScreenDC sdc; wxMemoryDC mdc; wxBitmap bmp(r.width, r.height,-1); //wxBitMap *bmp=wxEmptyImage(r.width,r.height); mdc.SelectObject(bmp); mdc.Blit((wxCoord)0, (wxCoord)0, (wxCoord)r.width, (wxCoord)r.height, &sdc, (wxCoord)r.x, (wxCoord)r.y); mdc.SelectObject(wxNullBitmap); wxDateTime d=wxDateTime::Now(); // wxDirDialog sfs(this,_("Choose a Directory")); //,wxT(""),wxT(""),style=wxFD_OPEN); wxString filename=wxSaveFileSelector(_("Please give a filename for the screenshot"),wxT("png"),wxT("Sleepyhead-")+d.Format(wxT("%Y%m%d-%H%M%S")),this); if (!filename.IsEmpty()) { if (!filename.Lower().EndsWith(wxT(".png"))) filename+=wxT(".png"); wxImage img=bmp.ConvertToImage(); if (!img.SaveFile(filename, wxBITMAP_TYPE_PNG)) { wxLogError(wxT("Couldn't save screenshot ")+filename); } } } void SleepyHeadFrame::OnShowSerial(wxCommandEvent& event) { pref["ShowSerialNumbers"]=event.IsChecked(); } void SleepyHeadFrame::OnAbout(wxCommandEvent &event) { wxString msg = wxbuildinfo(long_f); msg=wxTheApp->GetAppName()+wxT(" v")+wxString(AutoVersion::FULLVERSION_STRING,wxConvUTF8)+wxT("\nAuthors: Mark Watkins / Troy Schultz\nThis is alpha software is guaranteed to break regularly!\nUse at your own risk.\n\nLicense: GPL"); //,AutoVersion::DATE,AutoVersion::MONTH,AutoVersion::YEAR wxMessageBox(msg, _("Welcome to..."),0,this); } void SleepyHeadFrame::OnImportSD(wxCommandEvent &event) { wxDirDialog dd(this,_("Choose a Directory")); //,wxT(""),wxT(""),style=wxFD_OPEN); if (dd.ShowModal()!=wxID_OK) return; loader_progress=new wxProgressDialog(wxT("SleepyHead"),wxT("Please Wait..."),100,this, wxPD_APP_MODAL|wxPD_AUTO_HIDE|wxPD_SMOOTH); loader_progress->Hide(); wxString path=dd.GetPath(); loader_progress->Update(0); loader_progress->Show(); profile->Import(path); loader_progress->Update(100); loader_progress->Show(false); loader_progress->Destroy(); loader_progress=NULL; int idx=main_auinotebook->GetPageIndex(daily); if (idx!=wxNOT_FOUND) { daily->ResetDate(); daily->RefreshData(); daily->Refresh(); } idx=main_auinotebook->GetPageIndex(summary); if (idx!=wxNOT_FOUND) { summary->ResetProfile(profile); // resets the date ranges.. summary->RefreshData(); summary->Refresh(); } } void SleepyHeadFrame::OnViewMenuDaily( wxCommandEvent& event ) { int idx=main_auinotebook->GetPageIndex(daily); unsigned int id; if (idx==wxNOT_FOUND) { daily=new Daily(this,profile); main_auinotebook->AddPage(daily,_("Daily"),true); daily->RefreshData(); daily->Refresh(); } else { main_auinotebook->SetSelection(idx); } } void SleepyHeadFrame::OnViewMenuSummary( wxCommandEvent& event ) { int id,idx=main_auinotebook->GetPageIndex(summary); if (idx==wxNOT_FOUND) { summary=new Summary(this,profile); main_auinotebook->AddPage(summary,_("Summary"),true); summary->ResetProfile(profile); summary->RefreshData(); summary->Refresh(); } else { main_auinotebook->SetSelection(idx); } } Summary::Summary(wxWindow *win,Profile *_profile) :SummaryPanel(win),profile(_profile) { AddData(ahidata=new HistoryData(profile)); AddData(pressure=new HistoryCodeData(profile,CPAP_PressureAverage)); AddData(pressure_min=new HistoryCodeData(profile,CPAP_PressureMinAchieved)); AddData(pressure_max=new HistoryCodeData(profile,CPAP_PressureMaxAchieved)); AddData(pressure_eap=new HistoryCodeData(profile,BIPAP_EAPMax)); AddData(pressure_iap=new HistoryCodeData(profile,BIPAP_IAPMin)); AddData(leak=new HistoryCodeData(profile,CPAP_LeakMedian)); AddData(usage=new UsageHistoryData(profile,UHD_Hours)); AddData(waketime=new UsageHistoryData(profile,UHD_Waketime)); AddData(bedtime=new UsageHistoryData(profile,UHD_Bedtime)); AHI=new gGraphWindow(ScrolledWindow,-1,wxT("AHI"),wxPoint(0,0), wxSize(400,160), wxNO_BORDER); AHI->SetMargins(10,15,65,80); AHI->AddLayer(new gBarChart(ahidata,wxRED)); AHI->AddLayer(new gFooBar()); // AHI->AddLayer(new gXAxis(NULL,wxBLACK)); //AHI->AddLayer(new gLineChart(ahidata,wxRED)); fgSizer->Add(AHI,1,wxEXPAND); PRESSURE=new gGraphWindow(ScrolledWindow,-1,wxT("Pressure"),wxPoint(0,0), wxSize(400,160), wxNO_BORDER); PRESSURE->SetMargins(10,15,65,80); //PRESSURE->AddLayer(new gBarChart(pressure,wxBLUE)); //PRESSURE->AddLayer(new gLineChart(pressure_eap,wxRED,6192,false,true)); //PRESSURE->AddLayer(new gLineChart(pressure_iap,wxBLUE,6192,false,true)); PRESSURE->AddLayer(new gYAxis(wxBLACK)); PRESSURE->AddLayer(new gXAxis(wxBLACK)); PRESSURE->AddLayer(new gLineChart(pressure_max,wxBLUE,6192,false,true,true)); PRESSURE->AddLayer(new gLineChart(pressure_min,wxRED,6192,false,true,true)); // PRESSURE->AddLayer(new gLineChart(pressure_eap,wxPURPLE,6192,false,true,true)); //PRESSURE->AddLayer(new gLineChart(pressure_iap,wxYELLOW,6192,false,true,true)); PRESSURE->AddLayer(new gLineChart(pressure,wxDARK_GREEN,6192,false,true,true)); fgSizer->Add(PRESSURE,1,wxEXPAND); LEAK=new gGraphWindow(ScrolledWindow,-1,wxT("Mask Leak"),wxPoint(0,0), wxSize(400,160), wxNO_BORDER); LEAK->SetMargins(10,15,65,80); //LEAK->AddLayer(new gBarChart(leak,wxYELLOW)); LEAK->AddLayer(new gXAxis(wxBLACK)); LEAK->AddLayer(new gLineChart(leak,wxPURPLE,6192,false,false,true)); fgSizer->Add(LEAK,1,wxEXPAND); USAGE=new gGraphWindow(ScrolledWindow,-1,wxT("Usage (Hours)"),wxPoint(0,0), wxSize(400,160), wxNO_BORDER); USAGE->SetMargins(10,15,65,80); USAGE->AddLayer(new gBarChart(usage,wxGREEN)); USAGE->AddLayer(new gFooBar()); //USAGE->AddLayer(new gXAxis(wxBLACK)); //USAGE->AddLayer(new gLineChart(usage,wxGREEN)); fgSizer->Add(USAGE,1,wxEXPAND); // Logo.LoadFile(wxT("./pic.png")); //wxMemoryFSHandler::AddFile(_T("test.png"), Logo, wxBITMAP_TYPE_PNG); // RefreshData(); } Summary::~Summary() { // wxMemoryFSHandler::RemoveFile(_T("test.png")); } void Summary::ResetProfile(Profile *p) { profile=p; if (profile->FirstDay().IsValid()) { StartDatePicker->SetRange(profile->FirstDay()+wxTimeSpan::Day(),profile->LastDay()+wxTimeSpan::Day()); EndDatePicker->SetRange(profile->FirstDay()+wxTimeSpan::Day(),profile->LastDay()+wxTimeSpan::Day()); StartDatePicker->SetValue(profile->FirstDay()+wxTimeSpan::Day()); EndDatePicker->SetValue(profile->LastDay()+wxTimeSpan::Day()); for (list::iterator h=Data.begin();h!=Data.end();h++) { (*h)->SetProfile(p); (*h)->ResetDateRange(); } } } void Summary::RefreshData() { for (list::iterator h=Data.begin();h!=Data.end();h++) { (*h)->Update(); } wxString submodel=_("Unknown Model"); double ahi=ahidata->GetAverage(); double avp=pressure->GetAverage(); double apmin=pressure_min->GetAverage(); double apmax=pressure_max->GetAverage(); double aeap=pressure_eap->GetAverage(); double aiap=pressure_iap->GetAverage(); double bt=fmod(bedtime->GetAverage(),12.0); double ua=usage->GetAverage(); double wt=waketime->GetAverage(); //fmod(bt+ua,12.0); wxString html=wxT(""); //html=html+wxT(""); html=html+wxT("\n"); html=html+wxT("\n"); /*if (machine->properties.find(wxT("SubModel"))!=machine->properties.end()) submodel=wxT("
\n ")+machine->properties[wxT("SubModel")]; html=html+wxT("\n"); html=html+wxT(""); */ html=html+wxT("\n"); html=html+wxT("\n"); html=html+wxT("\n"); html=html+wxT("\n"); if (aiap>0) { html=html+wxT("\n"); html=html+wxT("\n"); } else { if (apmin!=apmax) { html=html+wxT("\n"); html=html+wxT("\n"); } } html=html+wxT("\n"); html=html+wxT("\n"); html=html+wxT("\n"); html=html+wxT("\n"); html=html+wxT("\n"); html=html+wxT("\n"); html=html+wxT("
")+_("Machine Information has been removed because this page has become machine agnostic. Not sure what to display here.")+wxT("
")+machine->properties[wxT("Brand")]+wxT("
")+machine->properties[wxT("Model")]+wxT(" ")+machine->properties[wxT("ModelNumber")]+submodel+wxT("
")+_("Firmware")+wxT(" ")+machine->properties[wxT("SoftwareVersion")]+wxT("
  
")+_("Indice Averages")+wxT("
")+_("AHI")+wxT("")+wxString::Format(wxT("%0.2f"),ahi)+wxT("
")+_("Pressure Avg")+wxT("")+wxString::Format(wxT("%0.2fcmH2O"),avp)+wxT("
")+_("IPAP Avg")+wxT("")+wxString::Format(wxT("%0.2fcmH2O"),aiap)+wxT("
")+_("EPAP Avg")+wxT("")+wxString::Format(wxT("%0.2fcmH2O"),aeap)+wxT("
")+_("Pressure Min")+wxT("")+wxString::Format(wxT("%0.2fcmH2O"),apmin)+wxT("
")+_("Pressure Max")+wxT("")+wxString::Format(wxT("%0.2fcmH2O"),apmax)+wxT("
  
")+_("Mask Leaks")+wxT("")+wxString::Format(wxT("%0.2f"),leak->GetAverage())+wxT("
  
")+_("Bedtime")+wxT("")+wxString::Format(wxT("%02.0f:%02i"),bt,int(bt*60) % 60)+wxT("
")+_("Waketime")+wxT("")+wxString::Format(wxT("%02.0f:%02i"),wt,int(wt*60) % 60)+wxT("
")+_("Hours/Night")+wxT("")+wxString::Format(wxT("%02.0f:%02i"),ua,int(ua*60)%60)+wxT("
"); html+=wxT(""); HTMLInfo->SetPage(html); } void Summary::EnableDatePickers(bool b) { StartDatePicker->Enable(b); sdLabel->Enable(b); EndDatePicker->Enable(b); edLabel->Enable(b); } void Summary::OnRBSelect( wxCommandEvent& event ) { wxDateTime start=StartDatePicker->GetValue(); wxDateTime end=EndDatePicker->GetValue(); if (rbCustomDate->GetValue()) { EnableDatePickers(true); } else if (rbAll->GetValue()) { start=profile->FirstDay()+wxTimeSpan::Day(); end=profile->LastDay()+wxTimeSpan::Day(); EnableDatePickers(false); } else if (rbLastMonth->GetValue()) { end=profile->LastDay()+wxTimeSpan::Day(); start=end-wxTimeSpan::Days(30-1); EnableDatePickers(false); } else if (rbLastWeek->GetValue()) { end=profile->LastDay()+wxTimeSpan::Day(); start=end-wxTimeSpan::Days(7-1); EnableDatePickers(false); } StartDatePicker->SetValue(start); EndDatePicker->SetValue(end); for (list::iterator h=Data.begin();h!=Data.end();h++) { (*h)->SetDateRange(start-wxTimeSpan::Day(),end); } } void Summary::OnStartDateChanged( wxDateEvent& event ) { wxDateTime start=StartDatePicker->GetValue()-wxTimeSpan::Days(2); wxDateTime end=EndDatePicker->GetValue()-wxTimeSpan::Day(); for (list::iterator h=Data.begin();h!=Data.end();h++) { (*h)->SetDateRange(start,end); } } void Summary::OnEndDateChanged( wxDateEvent& event ) { wxDateTime start=StartDatePicker->GetValue()-wxTimeSpan::Days(2); wxDateTime end=EndDatePicker->GetValue()-wxTimeSpan::Day(); for (list::iterator h=Data.begin();h!=Data.end();h++) { (*h)->SetDateRange(start,end); } } void Summary::OnClose(wxCloseEvent &event) { Destroy(); } /* MyListBox::MyListBox(wxWindow* parent, wxWindowID id, const wxPoint& pos, const wxSize& size, int n, const wxString choices[], long style, const wxValidator& validator, const wxString& name) :wxListBox(parent,id,pos,size,n,choices,style,validator,name) { InvalidateBestSize(); Clear(); } wxSize MyListBox::DoGetBestSize() const { wxSize best(200, 50); // CacheBestSize(best); return best; } */ Daily::Daily(wxWindow *win,Profile *p) :DailyPanel(win),profile(p) { //SessionList=new wxListBox(this, wxID_ANY, wxDefaultPosition, wxDefaultSize, 0, NULL, 0 ); //SessionList->SetMinSize(wxSize(200,50)); //SessionList->SetMaxSize(wxSize(200,50)); //m_mgr.AddPane(SessionList,wxLEFT,wxT("Sessions")); AddData(tap_eap=new TAPData(CPAP_EAP)); AddData(tap_iap=new TAPData(CPAP_IAP)); AddData(tap=new TAPData(CPAP_Pressure)); TAP=new gGraphWindow(ScrolledWindow,-1,wxT("Time@Pressure"),wxPoint(0,0), wxSize(600,60), wxNO_BORDER); TAP->SetMargins(20,15,5,50); TAP->AddLayer(new gCandleStick(tap)); TAP_IAP=new gGraphWindow(ScrolledWindow,-1,wxT("Time@IPAP"),wxPoint(0,0), wxSize(600,60), wxNO_BORDER); TAP_IAP->SetMargins(20,15,5,50); TAP_IAP->AddLayer(new gCandleStick(tap_iap)); TAP_EAP=new gGraphWindow(ScrolledWindow,-1,wxT("Time@EPAP"),wxPoint(0,0), wxSize(600,60), wxNO_BORDER); TAP_EAP->SetMargins(20,15,5,50); TAP_EAP->AddLayer(new gCandleStick(tap_eap)); G_AHI=new gGraphWindow(ScrolledWindow,-1,wxT("Event Breakdown"),wxPoint(0,0), wxSize(600,60), wxNO_BORDER); G_AHI->SetMargins(20,15,5,50); AddData(g_ahi=new AHIData()); gCandleStick *l=new gCandleStick(g_ahi); l->AddName(wxT("H")); l->AddName(wxT("OA")); l->AddName(wxT("CA")); l->AddName(wxT("RE")); l->AddName(wxT("FL")); l->AddName(wxT("CSR")); l->color.clear(); l->color.push_back(wxBLUE); l->color.push_back(wxAQUA); l->color.push_back(wxPURPLE); l->color.push_back(wxYELLOW); l->color.push_back(wxBLACK); l->color.push_back(wxGREEN2); G_AHI->AddLayer(l); AddData(leakdata=new PressureData(CPAP_Leak,0)); //leakdata->ForceMinY(0); //leakdata->ForceMaxY(120); LEAK=new gGraphWindow(ScrolledWindow,-1,wxT("Mask Leaks"),wxPoint(0,0), wxSize(600,130), wxNO_BORDER); LEAK->AddLayer(new gLineChart(leakdata,wxPURPLE,4096,false,false,true)); LEAK->AddLayer(new gXAxis(wxBLACK)); AddData(pressure_iap=new PressureData(CPAP_IAP)); AddData(pressure_eap=new PressureData(CPAP_EAP)); AddData(prd=new PressureData(CPAP_Pressure)); PRD=new gGraphWindow(ScrolledWindow,-1,wxT("Pressure"),wxPoint(0,0), wxSize(600,130), wxNO_BORDER); PRD->AddLayer(new gLineChart(prd,wxDARK_GREEN,4096,false,false,true)); PRD->AddLayer(new gLineChart(pressure_iap,wxBLUE,4096,false,true,true)); PRD->AddLayer(new gLineChart(pressure_eap,wxRED,4096,false,true,true)); PRD->AddLayer(new gXAxis(wxBLACK)); AddData(frw=new FlowData()); FRW=new gGraphWindow(ScrolledWindow,-1,wxT("Flow Rate"),wxPoint(0,0), wxSize(600,150), wxNO_BORDER); AddData(flags[0]=new FlagData(CPAP_CSR,7,1,0)); AddData(flags[1]=new FlagData(CPAP_ClearAirway,6)); AddData(flags[2]=new FlagData(CPAP_Obstructive,5)); AddData(flags[3]=new FlagData(CPAP_Hypopnea,4)); AddData(flags[4]=new FlagData(CPAP_FlowLimit,3)); AddData(flags[5]=new FlagData(CPAP_VSnore,2)); AddData(flags[6]=new FlagData(CPAP_RERA,1)); AddData(flags[7]=new FlagData(PRS1_PressurePulse,1)); AddData(flags[8]=new FlagData(PRS1_VSnore2,1)); AddData(flags[9]=new FlagData(PRS1_Unknown0E,1)); FRW->AddLayer(new gLineOverlayBar(flags[0],wxGREEN2,wxT("CSR"))); FRW->AddLayer(new gLineChart(frw,wxBLACK,200000,true)); FRW->AddLayer(new gLineOverlayBar(flags[7],wxRED,wxT("PR"),LOT_Dot)); FRW->AddLayer(new gLineOverlayBar(flags[6],wxYELLOW,wxT("RE"))); FRW->AddLayer(new gLineOverlayBar(flags[9],wxDARK_GREEN,wxT("U0E"))); FRW->AddLayer(new gLineOverlayBar(flags[5],wxRED,wxT("VS"))); FRW->AddLayer(new gLineOverlayBar(flags[4],wxBLACK,wxT("FL"))); FRW->AddLayer(new gLineOverlayBar(flags[3],wxBLUE,wxT("H"))); FRW->AddLayer(new gLineOverlayBar(flags[2],wxAQUA,wxT("OA"))); FRW->AddLayer(new gLineOverlayBar(flags[1],wxPURPLE,wxT("CA"))); FRW->AddLayer(new gXAxis(wxBLACK)); SF=new gGraphWindow(ScrolledWindow,-1,wxT("Event Flags"),wxPoint(0,0), wxSize(600,180), wxNO_BORDER); // SF->SetMargins(10,15,20,80); SF->LinkZoom(FRW); FRW->LinkZoom(SF); #if defined(__UNIX__) // SF->LinkZoom(PRD); // Uncomment to link in more graphs.. Too slow on windows. // SF->LinkZoom(LEAK); #endif const int sfc=9; SF->AddLayer(new gFlagsLine(flags[9],wxDARK_GREEN,wxT("U0E"),8,sfc)); SF->AddLayer(new gFlagsLine(flags[8],wxRED,wxT("VS2"),6,sfc)); SF->AddLayer(new gFlagsLine(flags[6],wxYELLOW,wxT("RE"),7,sfc)); SF->AddLayer(new gFlagsLine(flags[5],wxRED,wxT("VS"),5,sfc)); SF->AddLayer(new gFlagsLine(flags[4],wxBLACK,wxT("FL"),4,sfc)); SF->AddLayer(new gFlagsLine(flags[3],wxBLUE,wxT("H"),3,sfc)); SF->AddLayer(new gFlagsLine(flags[2],wxAQUA,wxT("OA"),2,sfc)); SF->AddLayer(new gFlagsLine(flags[1],wxPURPLE,wxT("CA"),1,sfc)); SF->AddLayer(new gFlagsLine(flags[0],wxGREEN2,wxT("CSR"),0,sfc)); SF->AddLayer(new gXAxis(wxBLACK)); SF->AddLayer(new gFooBar()); fgSizer->Add(SF,1,wxEXPAND); fgSizer->Add(FRW,1,wxEXPAND); fgSizer->Add(PRD,1,wxEXPAND); fgSizer->Add(LEAK,1,wxEXPAND); fgSizer->Add(G_AHI,1,wxEXPAND); fgSizer->Add(TAP,1,wxEXPAND); fgSizer->Add(TAP_IAP,1,wxEXPAND); fgSizer->Add(TAP_EAP,1,wxEXPAND); ResetDate(); } Daily::~Daily() { // delete SessionList; } void Daily::OnClose(wxCloseEvent &event) { Destroy(); } void Daily::ResetDate() { foobar_datehack=false; // this exists due to a wxGTK bug. // RefreshData(); wxDateTime date; if (profile->LastDay().IsValid()) { date=profile->LastDay()+wxTimeSpan::Day(); Calendar->SetDate(date); } else { Calendar->SetDate(wxDateTime::Today()); } wxCalendarEvent ev; ev.SetDate(date); OnCalendarMonth(ev); } void Daily::RefreshData() { wxDateTime date=Calendar->GetDate(); date.ResetTime(); date.SetHour(0); date-=wxTimeSpan::Days(1); Day *d=NULL; if (profile->daylist.find(date)!=profile->daylist.end()) { vector::iterator di; for (di=profile->daylist[date].begin();di!=profile->daylist[date].end();di++) { if ((*di)->machine_type()==MT_CPAP) { d=(*di); break; } } } UpdateGraphs(d); if (d) { CPAPMode mode=(CPAPMode)d->summary_max(CPAP_Mode); if (mode!=MODE_BIPAP) { TAP_EAP->Show(false); TAP_IAP->Show(false); TAP->Show(true); } else { TAP->Show(false); TAP_IAP->Show(true); TAP_EAP->Show(true); } fgSizer->Layout(); ScrolledWindow->FitInside(); PRTypes pr=(PRTypes)d->summary_max(CPAP_PressureReliefType); wxString epr=PressureReliefNames[pr]+wxString::Format(wxT(" x%i"),(int)d->summary_max(CPAP_PressureReliefSetting)); wxString modestr=CPAPModeNames[mode]; float ahi=(d->count(CPAP_Obstructive)+d->count(CPAP_Hypopnea)+d->count(CPAP_ClearAirway))/d->hours(); float csr=(100.0/d->hours())*(d->sum(CPAP_CSR)/3600.0); float oai=d->count(CPAP_Obstructive)/d->hours(); float hi=d->count(CPAP_Hypopnea)/d->hours(); float cai=d->count(CPAP_ClearAirway)/d->hours(); float rei=d->count(CPAP_RERA)/d->hours(); float vsi=d->count(CPAP_VSnore)/d->hours(); float fli=d->count(CPAP_FlowLimit)/d->hours(); float p90=d->percentile(CPAP_Pressure,0,0.9); float eap90=d->percentile(CPAP_EAP,0,0.9); float iap90=d->percentile(CPAP_IAP,0,0.9); wxString submodel=_("Unknown Model"); wxString html=wxT("\n"); html=html+wxT("\n"); if (d->machine->properties.find(wxT("SubModel"))!=d->machine->properties.end()) submodel=wxT("
")+d->machine->properties[wxT("SubModel")]; html=html+wxT("\n"); if (pref.Exists("ShowSerialNumbers") && pref["ShowSerialNumbers"]) { html=html+wxT("\n"); } html=html+wxT("\n"); html=html+wxT("\n"); html=html+wxT("\n"); html=html+wxT("\n"); html=html+wxT("\n"); html=html+wxT("\n"); html=html+wxT("\n"); html=html+wxT("\n"); html=html+wxT("\n"); html=html+wxT("\n"); html=html+wxT("\n"); html=html+wxT("\n"); html=html+wxT("\n"); html=html+wxT("\n"); html=html+wxT("\n"); html=html+wxT("\n"); html=html+wxT("\n"); html=html+wxT("\n"); // html=html+wxT("\n"); // html=html+wxT("\n"); html=html+wxT("\n"); html=html+wxT("\n"); if (mode==MODE_CPAP) { html=html+wxT("\n"); } else if (mode==MODE_APAP) { html=html+wxT("\n"); html=html+wxT("\n"); html=html+wxT("\n"); html=html+wxT("\n"); } else if (mode==MODE_BIPAP) { html=html+wxT("\n"); html=html+wxT("\n"); html=html+wxT("\n"); html=html+wxT("\n"); html=html+wxT("\n"); html=html+wxT("\n"); html=html+wxT("\n"); html=html+wxT("\n"); } html=html+wxT("\n"); html=html+wxT("\n"); if (d->summary_avg(CPAP_BrokenSummary)==1) { html=html+wxT("\n"); } else { html=html+wxT("\n"); if (mode==MODE_CPAP) { html=html+wxT("\n"); } else if (mode==MODE_APAP) { html=html+wxT("\n"); html=html+wxT("\n"); } else if (mode==MODE_BIPAP) { html=html+wxT("\n"); html=html+wxT("\n"); } html=html+wxT("\n"); html=html+wxT("\n"); // check HumidiferStatus.. wxString str; if (bool(d->summary_max(CPAP_HumidifierStatus))) { str=wxString::Format(wxT("x%i"),(int)d->summary_max(CPAP_HumidifierSetting)); } else str=wxT("No"); html=html+wxT("\n"); html=html+wxT("\n"); html=html+wxT("\n"); html=html+wxT("\n"); html=html+wxT("\n"); html=html+wxT("\n"); if (bool(d->summary_max(PRS1_SystemResistanceStatus))) { str=wxString::Format(wxT("x%i"),(int)d->summary_max(PRS1_SystemResistanceSetting)); } else str=wxT("No"); html=html+wxT("\n"); } html=html+wxT("\n"); html=html+wxT("\n"); //SessionList->Clear(); for (vector::iterator i=d->begin();i!=d->end();i++) { //wxString a=wxString::Format(wxT("Session %i"),(*i)->session()); //SessionList->Append(a); html=html+wxT("\n"); html=html+wxT("\n"); } //SessionList->SetSelection(0); html=html+wxT("
")+_("Machine Information")+wxT("
")+d->machine->properties[wxT("Brand")]+wxT("
")+d->machine->properties[wxT("Model")]+wxT(" ")+d->machine->properties[wxT("ModelNumber")]+submodel+wxT("
")+d->machine->properties[wxT("Serial")]+wxT("
  
")+_("Sleep Times")+wxT("
")+_("Date")+wxT("")+d->first().Format(wxT("%x"))+wxT("
")+_("Sleep")+wxT("")+d->first().Format(wxT("%H:%M"))+wxT("
")+_("Wake")+wxT("")+d->last().Format(wxT("%H:%M"))+wxT("
")+_("Total Time")+wxT("")+d->total_time().Format(wxT("%H:%M hours"))+wxT("
  
")+_("Indices")+wxT("
")+_("AHI")+wxT("")+wxString::Format(wxT("%0.2f"),ahi)+wxT("
")+_("Obstructive")+wxT("")+wxString::Format(wxT("%0.2f"),oai)+wxT("
")+_("Hypopnea")+wxT("")+wxString::Format(wxT("%0.2f"),hi)+wxT("
")+_("ClearAirway")+wxT("")+wxString::Format(wxT("%0.2f"),cai)+wxT("
")+_("RERA")+wxT("")+wxString::Format(wxT("%0.2f"),rei)+wxT("
")+_("FlowLimit")+wxT("")+wxString::Format(wxT("%0.2f"),fli)+wxT("
")+_("Vsnore")+wxT("")+wxString::Format(wxT("%0.2f"),vsi)+wxT("
")+_("CSR")+wxT("")+wxString::Format(wxT("%0.2f%%"),csr)+wxT("
  
")+_("Other Information")+wxT("
  
Session Informaton
")+_("Mode")+wxT("")+modestr+wxT("
")+_("Relief")+wxT("")+epr+wxT("
")+_("Pressure")+wxT("")+wxString::Format(wxT("%.1fcmH2O"),d->summary_min(CPAP_PressureMin))+wxT("
")+_("Avg Pressure")+wxT("")+wxString::Format(wxT("%.2fcmH2O"),d->summary_avg(CPAP_PressureAverage))+wxT("
")+_("Min Reached")+wxT("")+wxString::Format(wxT("%.1fcmH2O"),d->summary_min(CPAP_PressureMinAchieved))+wxT("
")+_("Max Reached")+wxT("")+wxString::Format(wxT("%.1fcmH2O"),d->summary_max(CPAP_PressureMaxAchieved))+wxT("
")+_("90% Pressure")+wxT("")+wxString::Format(wxT("%.1fcmH2O"),p90)+wxT("
")+_("Avg IPAP")+wxT("")+wxString::Format(wxT("%.2fcmH2O"),d->summary_avg(BIPAP_IAPAverage))+wxT("
")+_("Avg EPAP")+wxT("")+wxString::Format(wxT("%.1fcmH2O"),d->summary_avg(BIPAP_EAPAverage))+wxT("
")+_("Min IPAP")+wxT("")+wxString::Format(wxT("%.2fcmH2O"),d->summary_min(BIPAP_IAPMin))+wxT("
")+_("Max IPAP")+wxT("")+wxString::Format(wxT("%.1fcmH2O"),d->summary_max(BIPAP_IAPMax))+wxT("
")+_("Min EPAP")+wxT("")+wxString::Format(wxT("%.1fcmH2O"),d->summary_min(BIPAP_EAPMin))+wxT("
")+_("Max EPAP")+wxT("")+wxString::Format(wxT("%.1fcmH2O"),d->summary_max(BIPAP_EAPMax))+wxT("
")+_("90% IPAP")+wxT("")+wxString::Format(wxT("%.1fcmH2O"),iap90)+wxT("
")+_("90% EPAP")+wxT("")+wxString::Format(wxT("%.1fcmH2O"),eap90)+wxT("
")+_("Avg Leak")+wxT("")+wxString::Format(wxT("%.2f"),d->summary_avg(CPAP_LeakAverage))+wxT("
  
")+_("No System Settings Recorded")+wxT("
")+_("System Settings")+wxT("
")+_("Pressure")+wxT("")+wxString::Format(wxT("%.1fcmH2O"),d->summary_min(CPAP_PressureMin))+wxT("
")+_("Min Pressure")+wxT("")+wxString::Format(wxT("%.1fcmH2O"),d->summary_min(CPAP_PressureMin))+wxT("
")+_("Max Pressure")+wxT("")+wxString::Format(wxT("%.1fcmH2O"),d->summary_max(CPAP_PressureMax))+wxT("
")+_("IPAP Pressure")+wxT("")+wxString::Format(wxT("%.1fcmH2O"),d->summary_min(CPAP_PressureMin))+wxT("
")+_("EPAP Pressure")+wxT("")+wxString::Format(wxT("%.1fcmH2O"),d->summary_max(CPAP_PressureMax))+wxT("
")+_("Ramp-Time")+wxT("")+wxString::Format(wxT("%imin"),(int)d->summary_max(CPAP_RampTime))+wxT("
")+_("Ramp-Prs.")+wxT("")+wxString::Format(wxT("%.1fcmH2O"),d->summary_min(CPAP_RampStartingPressure))+wxT("
")+_("Humidifier")+wxT("")+str+wxT("
")+_("System-Lock")+wxT("")+(bool(d->summary_max(PRS1_SystemLockStatus)) ? _("On") : _("Off"))+wxT("
")+_("Auto-Off")+wxT("")+(bool(d->summary_max(PRS1_AutoOff)) ? _("On") : _("Off"))+wxT("
")+_("Mask-Alert")+wxT("")+(bool(d->summary_max(PRS1_MaskAlert)) ? _("On") : _("Off"))+wxT("
")+_("Show-AHI")+wxT("")+(bool(d->summary_max(PRS1_ShowAHI)) ? _("On") : _("Off"))+wxT("
")+_("Hose-Size")+wxT("")+(bool(d->summary_max(PRS1_HoseDiameter)) ? _("22mm") : _("15mm"))+wxT("
")+_("Sys-Resist.")+wxT("")+str+wxT("
  
")+_("Session Files")+wxT("
")+(*i)->first().Format(wxT("%d-%m-%Y %H:%M:%S"))+wxT(" ")+wxString::Format(wxT("%05i"),(*i)->session())+wxT("
")+(*i)->last().Format(wxT("%d-%m-%Y %H:%M:%S"))+wxT(" ")+wxString::Format(wxT("%05i"),(*i)->session())+wxT("
"); html+=wxT(""); HTMLInfo->SetPage(html); /* if (d->size()>1) { if (!SessionList->IsShown()) { SessionList->Show(true); SessionList->SetMinSize(wxSize(200,45)); SessionList->SetMaxSize(wxSize(200,45)); m_mgr.AddPane(SessionList,wxLEFT,wxEmptyString); m_mgr.GetPane(SessionList).MinSize(200,45); m_mgr.GetPane(SessionList).MaxSize(200,45); m_mgr.GetPane(SessionList).CaptionVisible(false); m_mgr.GetPane(Calendar).Position(0); m_mgr.GetPane(SessionList).Position(1); m_mgr.GetPane(HTMLInfo).Position(2); m_mgr.Update(); //Refresh(); } } else { if (SessionList->IsShown()) { m_mgr.DetachPane(SessionList); SessionList->Hide(); m_mgr.Update(); //Refresh(); } //m_mgr.Update(); } */ } else { HTMLInfo->SetPage(_("No CPAP Machine Data Available")); /*if (SessionList->IsShown()) { m_mgr.DetachPane(SessionList); SessionList->Hide(); m_mgr.Update(); //Refresh(); }*/ } } void Daily::OnSelectSession( wxCommandEvent& event ) { if (event.IsSelection()) { int sessid=event.GetSelection(); wxLogMessage(wxT("Selected:")+wxString::Format(wxT("%i"),sessid)); } } ///usr/local/bin/upx ./bin/Windows/SleepyHead void Daily::OnCalendarDay( wxCalendarEvent& event ) { if (foobar_datehack) { OnCalendarMonth(event); foobar_datehack=false; } RefreshData(); } void Daily::UpdateGraphs(Day *day) { //if (!day) return; if (day) { day->OpenEvents(); day->OpenWaveforms(); } for (list::iterator g=Data.begin();g!=Data.end();g++) { if (day==NULL) { (*g)->SetMinX(0); (*g)->SetMaxX(0); } (*g)->Update(day); } }; void Daily::OnCalendarMonth( wxCalendarEvent& event ) { wxDateTime et=event.GetDate(); if (!et.IsValid()) { foobar_datehack=true; return; } wxDateTime::Month m=et.GetMonth(); int y=et.GetYear(); static wxFont f=*wxNORMAL_FONT; //wxSystemSettings::GetFont(wxSYS_DEFAULT_GUI_FONT); f.SetWeight(wxBOLD); //if (!machine) return; for (int i=1;i<31;i++) { int j=wxDateTime::GetNumberOfDays(m,y); if (i>j) break; wxDateTime d(i,m,y,0,0,0,0); d-=wxTimeSpan::Days(1); if ((profile->daylist.find(d)!=profile->daylist.end())) { #if wxCHECK_VERSION(2,9,0) Calendar->Mark(i,true); #else wxCalendarDateAttr *a=new wxCalendarDateAttr(); a->SetFont(f); //wxNORM Calendar->SetAttr(i,a); #endif } else { #if wxCHECK_VERSION(2,9,0) Calendar->Mark(i,false); #else Calendar->ResetAttr(i); #endif // Calendar->SetAttr(i,NULL); } } }