mirror of
https://gitlab.com/pholy/OSCAR-code.git
synced 2025-04-04 18:20:42 +00:00
Initial commit
This commit is contained in:
commit
52c22f92f3
165
GUIFrame.cpp
Normal file
165
GUIFrame.cpp
Normal file
@ -0,0 +1,165 @@
|
||||
///////////////////////////////////////////////////////////////////////////
|
||||
// C++ code generated with wxFormBuilder (version May 5 2011)
|
||||
// http://www.wxformbuilder.org/
|
||||
//
|
||||
// PLEASE DO "NOT" EDIT THIS FILE!
|
||||
///////////////////////////////////////////////////////////////////////////
|
||||
|
||||
#include "wx/wxprec.h"
|
||||
|
||||
#ifdef __BORLANDC__
|
||||
#pragma hdrstop
|
||||
#endif //__BORLANDC__
|
||||
|
||||
#ifndef WX_PRECOMP
|
||||
#include <wx/wx.h>
|
||||
#endif //WX_PRECOMP
|
||||
|
||||
#include "GUIFrame.h"
|
||||
|
||||
///////////////////////////////////////////////////////////////////////////
|
||||
|
||||
GUIFrame::GUIFrame( wxWindow* parent, wxWindowID id, const wxString& title, const wxPoint& pos, const wxSize& size, long style ) : wxFrame( parent, id, title, pos, size, style )
|
||||
{
|
||||
this->SetSizeHints( wxDefaultSize, wxDefaultSize );
|
||||
m_mgr.SetManagedWindow(this);
|
||||
|
||||
menubar = new wxMenuBar( 0 );
|
||||
FileMenu = new wxMenu();
|
||||
wxMenuItem* FileMenuImportSD;
|
||||
FileMenuImportSD = new wxMenuItem( FileMenu, wxID_ANY, wxString( _("&Import SD") ) , wxEmptyString, wxITEM_NORMAL );
|
||||
FileMenu->Append( FileMenuImportSD );
|
||||
|
||||
wxMenuItem* FileMenuPreferences;
|
||||
FileMenuPreferences = new wxMenuItem( FileMenu, wxID_ANY, wxString( _("&Preferences") ) , wxEmptyString, wxITEM_NORMAL );
|
||||
FileMenu->Append( FileMenuPreferences );
|
||||
|
||||
wxMenuItem* m_separator1;
|
||||
m_separator1 = FileMenu->AppendSeparator();
|
||||
|
||||
wxMenuItem* FileMenuExit;
|
||||
FileMenuExit = new wxMenuItem( FileMenu, wxID_QUIT, wxString( _("E&xit") ) , wxEmptyString, wxITEM_NORMAL );
|
||||
FileMenu->Append( FileMenuExit );
|
||||
|
||||
menubar->Append( FileMenu, _("&File") );
|
||||
|
||||
ViewMenu = new wxMenu();
|
||||
wxMenuItem* ViewMenuSummary;
|
||||
ViewMenuSummary = new wxMenuItem( ViewMenu, wxID_ANY, wxString( _("&Summary") ) , wxEmptyString, wxITEM_NORMAL );
|
||||
ViewMenu->Append( ViewMenuSummary );
|
||||
|
||||
wxMenuItem* ViewMenuDaily;
|
||||
ViewMenuDaily = new wxMenuItem( ViewMenu, wxID_ANY, wxString( _("&Daily") ) , wxEmptyString, wxITEM_NORMAL );
|
||||
ViewMenu->Append( ViewMenuDaily );
|
||||
|
||||
menubar->Append( ViewMenu, _("&View") );
|
||||
|
||||
HelpMenu = new wxMenu();
|
||||
wxMenuItem* HelpMenuAbout;
|
||||
HelpMenuAbout = new wxMenuItem( HelpMenu, wxID_ANY, wxString( _("&About") ) , wxEmptyString, wxITEM_NORMAL );
|
||||
HelpMenu->Append( HelpMenuAbout );
|
||||
|
||||
menubar->Append( HelpMenu, _("&Help") );
|
||||
|
||||
this->SetMenuBar( menubar );
|
||||
|
||||
statusBar = this->CreateStatusBar( 2, wxST_SIZEGRIP, wxID_ANY );
|
||||
main_auinotebook = new wxAuiNotebook( this, wxID_ANY, wxDefaultPosition, wxDefaultSize, wxAUI_NB_DEFAULT_STYLE );
|
||||
m_mgr.AddPane( main_auinotebook, wxAuiPaneInfo() .Center() .MaximizeButton( false ).MinimizeButton( false ).PinButton( true ).Dock().Resizable().FloatingSize( wxDefaultSize ).DockFixed( false ) );
|
||||
|
||||
|
||||
|
||||
m_mgr.Update();
|
||||
|
||||
// Connect Events
|
||||
this->Connect( wxEVT_CLOSE_WINDOW, wxCloseEventHandler( GUIFrame::OnClose ) );
|
||||
this->Connect( FileMenuImportSD->GetId(), wxEVT_COMMAND_MENU_SELECTED, wxCommandEventHandler( GUIFrame::OnImportSD ) );
|
||||
this->Connect( FileMenuPreferences->GetId(), wxEVT_COMMAND_MENU_SELECTED, wxCommandEventHandler( GUIFrame::OnPreferencesClicked ) );
|
||||
this->Connect( FileMenuExit->GetId(), wxEVT_COMMAND_MENU_SELECTED, wxCommandEventHandler( GUIFrame::OnQuit ) );
|
||||
this->Connect( ViewMenuSummary->GetId(), wxEVT_COMMAND_MENU_SELECTED, wxCommandEventHandler( GUIFrame::OnViewMenuSummary ) );
|
||||
this->Connect( ViewMenuDaily->GetId(), wxEVT_COMMAND_MENU_SELECTED, wxCommandEventHandler( GUIFrame::OnViewMenuDaily ) );
|
||||
this->Connect( HelpMenuAbout->GetId(), wxEVT_COMMAND_MENU_SELECTED, wxCommandEventHandler( GUIFrame::OnAbout ) );
|
||||
}
|
||||
|
||||
GUIFrame::~GUIFrame()
|
||||
{
|
||||
// Disconnect Events
|
||||
this->Disconnect( wxEVT_CLOSE_WINDOW, wxCloseEventHandler( GUIFrame::OnClose ) );
|
||||
this->Disconnect( wxID_ANY, wxEVT_COMMAND_MENU_SELECTED, wxCommandEventHandler( GUIFrame::OnImportSD ) );
|
||||
this->Disconnect( wxID_ANY, wxEVT_COMMAND_MENU_SELECTED, wxCommandEventHandler( GUIFrame::OnPreferencesClicked ) );
|
||||
this->Disconnect( wxID_QUIT, wxEVT_COMMAND_MENU_SELECTED, wxCommandEventHandler( GUIFrame::OnQuit ) );
|
||||
this->Disconnect( wxID_ANY, wxEVT_COMMAND_MENU_SELECTED, wxCommandEventHandler( GUIFrame::OnViewMenuSummary ) );
|
||||
this->Disconnect( wxID_ANY, wxEVT_COMMAND_MENU_SELECTED, wxCommandEventHandler( GUIFrame::OnViewMenuDaily ) );
|
||||
this->Disconnect( wxID_ANY, wxEVT_COMMAND_MENU_SELECTED, wxCommandEventHandler( GUIFrame::OnAbout ) );
|
||||
|
||||
m_mgr.UnInit();
|
||||
|
||||
}
|
||||
|
||||
DailyPanel::DailyPanel( wxWindow* parent, wxWindowID id, const wxPoint& pos, const wxSize& size, long style ) : wxPanel( parent, id, pos, size, style )
|
||||
{
|
||||
m_mgr.SetManagedWindow(this);
|
||||
|
||||
HTMLInfo = new wxHtmlWindow( this, wxID_ANY, wxDefaultPosition, wxDefaultSize, wxHW_SCROLLBAR_AUTO );
|
||||
m_mgr.AddPane( HTMLInfo, wxAuiPaneInfo() .Left() .Caption( wxT("Day Summary") ).CloseButton( false ).MaximizeButton( false ).MinimizeButton( false ).PinButton( true ).Dock().Resizable().FloatingSize( wxSize( 200,424 ) ).DockFixed( false ).Row( 0 ).Position( 1 ).MinSize( wxSize( 200,200 ) ).Layer( 0 ) );
|
||||
|
||||
ScrolledWindow = new wxScrolledWindow( this, wxID_ANY, wxDefaultPosition, wxDefaultSize, wxHSCROLL|wxVSCROLL );
|
||||
ScrolledWindow->SetScrollRate( 5, 5 );
|
||||
m_mgr.AddPane( ScrolledWindow, wxAuiPaneInfo() .Center() .Caption( wxT("Daily Information") ).CloseButton( false ).MaximizeButton( false ).MinimizeButton( false ).PinButton( true ).Dock().Resizable().FloatingSize( wxSize( -1,-1 ) ).Row( 0 ).Layer( 1 ).CentrePane() );
|
||||
|
||||
fgSizer = new wxFlexGridSizer( 0, 1, 0, 0 );
|
||||
fgSizer->AddGrowableCol( 0 );
|
||||
fgSizer->SetFlexibleDirection( wxVERTICAL );
|
||||
fgSizer->SetNonFlexibleGrowMode( wxFLEX_GROWMODE_ALL );
|
||||
|
||||
ScrolledWindow->SetSizer( fgSizer );
|
||||
ScrolledWindow->Layout();
|
||||
fgSizer->Fit( ScrolledWindow );
|
||||
Calendar = new wxCalendarCtrl( this, wxID_ANY, wxDefaultDateTime, wxDefaultPosition, wxDefaultSize, wxCAL_MONDAY_FIRST|wxCAL_SEQUENTIAL_MONTH_SELECTION|wxCAL_SHOW_SURROUNDING_WEEKS );
|
||||
m_mgr.AddPane( Calendar, wxAuiPaneInfo() .Left() .Caption( wxT("Selected Day") ).CloseButton( false ).MaximizeButton( false ).MinimizeButton( false ).PinButton( true ).PaneBorder( false ).Dock().Fixed().BottomDockable( false ).TopDockable( false ) );
|
||||
|
||||
|
||||
m_mgr.Update();
|
||||
|
||||
// Connect Events
|
||||
Calendar->Connect( wxEVT_CALENDAR_MONTH_CHANGED, wxCalendarEventHandler( DailyPanel::OnCalendarMonth ), NULL, this );
|
||||
Calendar->Connect( wxEVT_CALENDAR_SEL_CHANGED, wxCalendarEventHandler( DailyPanel::OnCalendarDay ), NULL, this );
|
||||
}
|
||||
|
||||
DailyPanel::~DailyPanel()
|
||||
{
|
||||
// Disconnect Events
|
||||
Calendar->Disconnect( wxEVT_CALENDAR_MONTH_CHANGED, wxCalendarEventHandler( DailyPanel::OnCalendarMonth ), NULL, this );
|
||||
Calendar->Disconnect( wxEVT_CALENDAR_SEL_CHANGED, wxCalendarEventHandler( DailyPanel::OnCalendarDay ), NULL, this );
|
||||
|
||||
m_mgr.UnInit();
|
||||
|
||||
}
|
||||
|
||||
SummaryPanel::SummaryPanel( wxWindow* parent, wxWindowID id, const wxPoint& pos, const wxSize& size, long style ) : wxPanel( parent, id, pos, size, style )
|
||||
{
|
||||
m_mgr.SetManagedWindow(this);
|
||||
|
||||
HTMLInfo = new wxHtmlWindow( this, wxID_ANY, wxDefaultPosition, wxDefaultSize, wxHW_SCROLLBAR_AUTO );
|
||||
m_mgr.AddPane( HTMLInfo, wxAuiPaneInfo() .Right() .Caption( wxT("Information") ).CloseButton( false ).MaximizeButton( false ).MinimizeButton( false ).PinButton( true ).Dock().Resizable().FloatingSize( wxDefaultSize ).DockFixed( false ).MinSize( wxSize( 200,400 ) ) );
|
||||
|
||||
ScrolledWindow = new wxScrolledWindow( this, wxID_ANY, wxDefaultPosition, wxDefaultSize, wxHSCROLL|wxVSCROLL );
|
||||
ScrolledWindow->SetScrollRate( 5, 5 );
|
||||
m_mgr.AddPane( ScrolledWindow, wxAuiPaneInfo() .Center() .Caption( wxT("Overview") ).CloseButton( false ).MaximizeButton( false ).MinimizeButton( false ).PinButton( true ).Dock().Resizable().FloatingSize( wxDefaultSize ).DockFixed( false ).MinSize( wxSize( 440,400 ) ) );
|
||||
|
||||
fgSizer = new wxFlexGridSizer( 0, 1, 0, 0 );
|
||||
fgSizer->SetFlexibleDirection( wxVERTICAL );
|
||||
fgSizer->SetNonFlexibleGrowMode( wxFLEX_GROWMODE_ALL );
|
||||
|
||||
ScrolledWindow->SetSizer( fgSizer );
|
||||
ScrolledWindow->Layout();
|
||||
fgSizer->Fit( ScrolledWindow );
|
||||
|
||||
m_mgr.Update();
|
||||
}
|
||||
|
||||
SummaryPanel::~SummaryPanel()
|
||||
{
|
||||
m_mgr.UnInit();
|
||||
|
||||
}
|
116
GUIFrame.h
Normal file
116
GUIFrame.h
Normal file
@ -0,0 +1,116 @@
|
||||
///////////////////////////////////////////////////////////////////////////
|
||||
// C++ code generated with wxFormBuilder (version May 5 2011)
|
||||
// http://www.wxformbuilder.org/
|
||||
//
|
||||
// PLEASE DO "NOT" EDIT THIS FILE!
|
||||
///////////////////////////////////////////////////////////////////////////
|
||||
|
||||
#ifndef __GUIFrame__
|
||||
#define __GUIFrame__
|
||||
|
||||
#include <wx/intl.h>
|
||||
|
||||
#include <wx/string.h>
|
||||
#include <wx/bitmap.h>
|
||||
#include <wx/image.h>
|
||||
#include <wx/icon.h>
|
||||
#include <wx/menu.h>
|
||||
#include <wx/gdicmn.h>
|
||||
#include <wx/font.h>
|
||||
#include <wx/colour.h>
|
||||
#include <wx/settings.h>
|
||||
#include <wx/statusbr.h>
|
||||
#include <wx/aui/auibook.h>
|
||||
#include <wx/frame.h>
|
||||
#include <wx/aui/aui.h>
|
||||
#include <wx/html/htmlwin.h>
|
||||
#include <wx/sizer.h>
|
||||
#include <wx/scrolwin.h>
|
||||
#include <wx/calctrl.h>
|
||||
#include <wx/panel.h>
|
||||
|
||||
///////////////////////////////////////////////////////////////////////////
|
||||
|
||||
#define wxID_QUIT 1000
|
||||
|
||||
///////////////////////////////////////////////////////////////////////////////
|
||||
/// Class GUIFrame
|
||||
///////////////////////////////////////////////////////////////////////////////
|
||||
class GUIFrame : public wxFrame
|
||||
{
|
||||
private:
|
||||
|
||||
protected:
|
||||
wxMenuBar* menubar;
|
||||
wxMenu* FileMenu;
|
||||
wxMenu* ViewMenu;
|
||||
wxMenu* HelpMenu;
|
||||
wxStatusBar* statusBar;
|
||||
|
||||
// Virtual event handlers, overide them in your derived class
|
||||
virtual void OnClose( wxCloseEvent& event ) { event.Skip(); }
|
||||
virtual void OnImportSD( wxCommandEvent& event ) { event.Skip(); }
|
||||
virtual void OnPreferencesClicked( wxCommandEvent& event ) { event.Skip(); }
|
||||
virtual void OnQuit( wxCommandEvent& event ) { event.Skip(); }
|
||||
virtual void OnViewMenuSummary( wxCommandEvent& event ) { event.Skip(); }
|
||||
virtual void OnViewMenuDaily( wxCommandEvent& event ) { event.Skip(); }
|
||||
virtual void OnAbout( wxCommandEvent& event ) { event.Skip(); }
|
||||
|
||||
|
||||
public:
|
||||
wxAuiNotebook* main_auinotebook;
|
||||
|
||||
GUIFrame( wxWindow* parent, wxWindowID id = wxID_ANY, const wxString& title = _("SleepyHead"), const wxPoint& pos = wxDefaultPosition, const wxSize& size = wxSize( 1157,703 ), long style = wxDEFAULT_FRAME_STYLE|wxTAB_TRAVERSAL );
|
||||
wxAuiManager m_mgr;
|
||||
|
||||
~GUIFrame();
|
||||
|
||||
};
|
||||
|
||||
///////////////////////////////////////////////////////////////////////////////
|
||||
/// Class DailyPanel
|
||||
///////////////////////////////////////////////////////////////////////////////
|
||||
class DailyPanel : public wxPanel
|
||||
{
|
||||
private:
|
||||
|
||||
protected:
|
||||
wxHtmlWindow* HTMLInfo;
|
||||
wxScrolledWindow* ScrolledWindow;
|
||||
wxFlexGridSizer* fgSizer;
|
||||
wxCalendarCtrl* Calendar;
|
||||
|
||||
// Virtual event handlers, overide them in your derived class
|
||||
virtual void OnCalendarMonth( wxCalendarEvent& event ) { event.Skip(); }
|
||||
virtual void OnCalendarDay( wxCalendarEvent& event ) { event.Skip(); }
|
||||
|
||||
|
||||
public:
|
||||
|
||||
DailyPanel( wxWindow* parent, wxWindowID id = wxID_ANY, const wxPoint& pos = wxDefaultPosition, const wxSize& size = wxSize( 935,573 ), long style = wxTAB_TRAVERSAL ); wxAuiManager m_mgr;
|
||||
|
||||
~DailyPanel();
|
||||
|
||||
};
|
||||
|
||||
///////////////////////////////////////////////////////////////////////////////
|
||||
/// Class SummaryPanel
|
||||
///////////////////////////////////////////////////////////////////////////////
|
||||
class SummaryPanel : public wxPanel
|
||||
{
|
||||
private:
|
||||
|
||||
protected:
|
||||
wxHtmlWindow* HTMLInfo;
|
||||
wxScrolledWindow* ScrolledWindow;
|
||||
wxFlexGridSizer* fgSizer;
|
||||
|
||||
public:
|
||||
|
||||
SummaryPanel( wxWindow* parent, wxWindowID id = wxID_ANY, const wxPoint& pos = wxDefaultPosition, const wxSize& size = wxSize( 712,445 ), long style = wxTAB_TRAVERSAL ); wxAuiManager m_mgr;
|
||||
|
||||
~SummaryPanel();
|
||||
|
||||
};
|
||||
|
||||
#endif //__GUIFrame__
|
106
SleepyHead.cbp
Normal file
106
SleepyHead.cbp
Normal file
@ -0,0 +1,106 @@
|
||||
<?xml version="1.0" encoding="UTF-8" standalone="yes" ?>
|
||||
<CodeBlocks_project_file>
|
||||
<FileVersion major="1" minor="6" />
|
||||
<Project>
|
||||
<Option title="SleepyHead" />
|
||||
<Option pch_mode="2" />
|
||||
<Option compiler="gcc" />
|
||||
<Build>
|
||||
<Target title="Debug">
|
||||
<Option output="bin/Debug/SleepyHead" prefix_auto="1" extension_auto="1" />
|
||||
<Option object_output="obj/Debug/" />
|
||||
<Option type="0" />
|
||||
<Option compiler="gcc" />
|
||||
<Option projectLinkerOptionsRelation="2" />
|
||||
<Compiler>
|
||||
<Add option="-march=core2" />
|
||||
<Add option="-g" />
|
||||
</Compiler>
|
||||
</Target>
|
||||
<Target title="Release">
|
||||
<Option output="bin/Release/SleepyHead" prefix_auto="1" extension_auto="1" />
|
||||
<Option object_output="obj/Release/" />
|
||||
<Option type="0" />
|
||||
<Option compiler="gcc" />
|
||||
<Option projectLinkerOptionsRelation="2" />
|
||||
<Compiler>
|
||||
<Add option="-march=core2" />
|
||||
<Add option="-O2" />
|
||||
<Add option="-pg" />
|
||||
</Compiler>
|
||||
<Linker>
|
||||
<Add option="-pg" />
|
||||
</Linker>
|
||||
</Target>
|
||||
<Target title="Windows">
|
||||
<Option output="bin/Windows/SleepyHead" prefix_auto="1" extension_auto="1" />
|
||||
<Option working_dir="bin/Windows/" />
|
||||
<Option object_output="obj/Windows/" />
|
||||
<Option type="0" />
|
||||
<Option compiler="mingw" />
|
||||
<Option projectCompilerOptionsRelation="1" />
|
||||
<Option projectLinkerOptionsRelation="1" />
|
||||
<Compiler>
|
||||
<Add option="-march=i686" />
|
||||
<Add option="-O2" />
|
||||
<Add option="`i686-pc-mingw32-wx-config --version=2.8 --cflags`" />
|
||||
<Add option="-std=gnu++0x" />
|
||||
</Compiler>
|
||||
<Linker>
|
||||
<Add option="-s" />
|
||||
<Add option="`i686-pc-mingw32-wx-config --version=2.8 --libs richtext,aui,xrc,qa,html,adv,core,xml,net,base`" />
|
||||
</Linker>
|
||||
</Target>
|
||||
</Build>
|
||||
<Compiler>
|
||||
<Add option="-Wall" />
|
||||
<Add option="`wx-config --version=2.8 --cflags`" />
|
||||
<Add option="-std=gnu++0x" />
|
||||
<Add directory="libs" />
|
||||
<Add directory="graphs" />
|
||||
</Compiler>
|
||||
<Linker>
|
||||
<Add option="`wx-config --version=2.8 --libs richtext,aui,xrc,qa,html,adv,core,xml,net,base`" />
|
||||
</Linker>
|
||||
<Unit filename="GUIFrame.cpp" />
|
||||
<Unit filename="GUIFrame.h" />
|
||||
<Unit filename="SleepyHeadApp.cpp" />
|
||||
<Unit filename="SleepyHeadApp.h" />
|
||||
<Unit filename="SleepyHeadMain.cpp" />
|
||||
<Unit filename="SleepyHeadMain.h" />
|
||||
<Unit filename="WxWizFrame.fbp" />
|
||||
<Unit filename="graphs/graph.cpp" />
|
||||
<Unit filename="graphs/graph.h" />
|
||||
<Unit filename="libs/sleeplib/binary_file.cpp" />
|
||||
<Unit filename="libs/sleeplib/binary_file.h" />
|
||||
<Unit filename="libs/sleeplib/machine.cpp" />
|
||||
<Unit filename="libs/sleeplib/machine.h" />
|
||||
<Unit filename="libs/sleeplib/machine_loader.cpp" />
|
||||
<Unit filename="libs/sleeplib/machine_loader.h" />
|
||||
<Unit filename="libs/sleeplib/preferences.cpp" />
|
||||
<Unit filename="libs/sleeplib/preferences.h" />
|
||||
<Unit filename="libs/sleeplib/profiles.cpp" />
|
||||
<Unit filename="libs/sleeplib/profiles.h" />
|
||||
<Unit filename="libs/sleeplib/prs1_loader.cpp" />
|
||||
<Unit filename="libs/sleeplib/prs1_loader.h" />
|
||||
<Unit filename="libs/tinyxml/tinystr.cpp" />
|
||||
<Unit filename="libs/tinyxml/tinystr.h" />
|
||||
<Unit filename="libs/tinyxml/tinyxml.cpp" />
|
||||
<Unit filename="libs/tinyxml/tinyxml.h" />
|
||||
<Unit filename="libs/tinyxml/tinyxmlerror.cpp" />
|
||||
<Unit filename="libs/tinyxml/tinyxmlparser.cpp" />
|
||||
<Unit filename="version.h" />
|
||||
<Extensions>
|
||||
<code_completion />
|
||||
<envvars />
|
||||
<debugger />
|
||||
<lib_finder disable_auto="1" />
|
||||
<AutoVersioning>
|
||||
<Scheme minor_max="10" build_max="0" rev_max="0" rev_rand_max="10" build_times_to_increment_minor="300" />
|
||||
<Settings autoincrement="0" date_declarations="1" do_auto_increment="1" ask_to_increment="0" language="C++" svn="0" svn_directory="" header_path="version.h" />
|
||||
<Changes_Log show_changes_editor="0" app_title="released version %M.%m.%b of %p" changeslog_path="ChangesLog.txt" />
|
||||
<Code header_guard="VERSION_H" namespace="AutoVersion" prefix="" />
|
||||
</AutoVersioning>
|
||||
</Extensions>
|
||||
</Project>
|
||||
</CodeBlocks_project_file>
|
7787
SleepyHead.depend
Normal file
7787
SleepyHead.depend
Normal file
File diff suppressed because it is too large
Load Diff
64
SleepyHead.layout
Normal file
64
SleepyHead.layout
Normal file
@ -0,0 +1,64 @@
|
||||
<?xml version="1.0" encoding="UTF-8" standalone="yes" ?>
|
||||
<CodeBlocks_layout_file>
|
||||
<ActiveTarget name="Windows" />
|
||||
<File name="GUIFrame.cpp" open="0" top="0" tabpos="7">
|
||||
<Cursor position="1774" topLine="0" />
|
||||
</File>
|
||||
<File name="SleepyHeadApp.cpp" open="1" top="0" tabpos="12">
|
||||
<Cursor position="308" topLine="0" />
|
||||
</File>
|
||||
<File name="SleepyHeadMain.cpp" open="1" top="0" tabpos="3">
|
||||
<Cursor position="202" topLine="0" />
|
||||
</File>
|
||||
<File name="SleepyHeadMain.h" open="1" top="0" tabpos="4">
|
||||
<Cursor position="199" topLine="0" />
|
||||
</File>
|
||||
<File name="graphs/graph.cpp" open="1" top="0" tabpos="6">
|
||||
<Cursor position="112" topLine="0" />
|
||||
</File>
|
||||
<File name="graphs/graph.h" open="1" top="0" tabpos="5">
|
||||
<Cursor position="0" topLine="0" />
|
||||
</File>
|
||||
<File name="libs/sleeplib/binary_file.cpp" open="1" top="0" tabpos="11">
|
||||
<Cursor position="0" topLine="0" />
|
||||
</File>
|
||||
<File name="libs/sleeplib/binary_file.h" open="1" top="0" tabpos="14">
|
||||
<Cursor position="0" topLine="0" />
|
||||
</File>
|
||||
<File name="libs/sleeplib/machine.cpp" open="1" top="0" tabpos="1">
|
||||
<Cursor position="4763" topLine="174" />
|
||||
</File>
|
||||
<File name="libs/sleeplib/machine.h" open="1" top="0" tabpos="2">
|
||||
<Cursor position="108" topLine="0" />
|
||||
</File>
|
||||
<File name="libs/sleeplib/machine_loader.cpp" open="1" top="0" tabpos="8">
|
||||
<Cursor position="0" topLine="0" />
|
||||
</File>
|
||||
<File name="libs/sleeplib/machine_loader.h" open="1" top="0" tabpos="15">
|
||||
<Cursor position="0" topLine="0" />
|
||||
</File>
|
||||
<File name="libs/sleeplib/preferences.cpp" open="1" top="0" tabpos="9">
|
||||
<Cursor position="0" topLine="0" />
|
||||
</File>
|
||||
<File name="libs/sleeplib/preferences.h" open="1" top="0" tabpos="16">
|
||||
<Cursor position="31" topLine="0" />
|
||||
</File>
|
||||
<File name="libs/sleeplib/profiles.cpp" open="1" top="0" tabpos="10">
|
||||
<Cursor position="22" topLine="0" />
|
||||
</File>
|
||||
<File name="libs/sleeplib/profiles.h" open="1" top="0" tabpos="17">
|
||||
<Cursor position="28" topLine="0" />
|
||||
</File>
|
||||
<File name="libs/sleeplib/prs1_loader.cpp" open="1" top="0" tabpos="7">
|
||||
<Cursor position="0" topLine="0" />
|
||||
</File>
|
||||
<File name="libs/sleeplib/prs1_loader.h" open="1" top="1" tabpos="18">
|
||||
<Cursor position="31" topLine="0" />
|
||||
</File>
|
||||
<File name="libs/tinyxml/tinystr.cpp" open="1" top="0" tabpos="13">
|
||||
<Cursor position="0" topLine="0" />
|
||||
</File>
|
||||
<File name="version.h" open="0" top="0" tabpos="11">
|
||||
<Cursor position="0" topLine="0" />
|
||||
</File>
|
||||
</CodeBlocks_layout_file>
|
67
SleepyHeadApp.cpp
Normal file
67
SleepyHeadApp.cpp
Normal file
@ -0,0 +1,67 @@
|
||||
/***************************************************************
|
||||
* Name: SleepyHeadApp.cpp
|
||||
* Purpose: Code for Application Class
|
||||
* Author: Mark Watkins (jedimark64@gmail.com)
|
||||
* 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 <iostream>
|
||||
|
||||
#include <wx/log.h>
|
||||
#include <wx/progdlg.h>
|
||||
#include <wx/datetime.h>
|
||||
#include "SleepyHeadApp.h"
|
||||
#include "SleepyHeadMain.h"
|
||||
#include "version.h"
|
||||
|
||||
#include "sleeplib/prs1_loader.h"
|
||||
#include "sleeplib/profiles.h"
|
||||
|
||||
IMPLEMENT_APP(SleepyHeadApp);
|
||||
|
||||
bool SleepyHeadApp::OnInit()
|
||||
{
|
||||
// Initialize the logger
|
||||
|
||||
wxLog *logger=new wxLogStream(&std::cout);
|
||||
wxLog::SetActiveTarget(logger);
|
||||
|
||||
//wxDateTime::SetCountry(wxDateTime::USA);
|
||||
SetAppName(_("SleepyHead"));
|
||||
PRS1Loader::Register();
|
||||
Profiles::Scan();
|
||||
|
||||
//loader_progress->Show();
|
||||
|
||||
pref["AppName"]=wxT("Awesome Sleep Application");
|
||||
pref["Version"]=wxString(AutoVersion::FULLVERSION_STRING,wxConvUTF8);
|
||||
|
||||
pref["Profile"]=wxGetUserId();
|
||||
profile=Profiles::Get(pref["Profile"]);
|
||||
|
||||
profile->LoadMachineData();
|
||||
// loader_progress->Show(false);
|
||||
|
||||
SleepyHeadFrame* frame = new SleepyHeadFrame(0L);
|
||||
|
||||
frame->Show();
|
||||
|
||||
return true;
|
||||
}
|
||||
|
||||
int SleepyHeadApp::OnExit()
|
||||
{
|
||||
//delete loader_progress;
|
||||
Profiles::Done();
|
||||
return true;
|
||||
}
|
22
SleepyHeadApp.h
Normal file
22
SleepyHeadApp.h
Normal file
@ -0,0 +1,22 @@
|
||||
/***************************************************************
|
||||
* Name: SleepyHeadApp.h
|
||||
* Purpose: Defines Application Class
|
||||
* Author: Mark Watkins (jedimark64@gmail.com)
|
||||
* Created: 2011-05-20
|
||||
* Copyright: Mark Watkins (http://sourceforge.net/projects/sleepyhead/)
|
||||
* License:
|
||||
**************************************************************/
|
||||
|
||||
#ifndef SLEEPYHEADAPP_H
|
||||
#define SLEEPYHEADAPP_H
|
||||
|
||||
#include <wx/app.h>
|
||||
|
||||
class SleepyHeadApp : public wxApp
|
||||
{
|
||||
public:
|
||||
virtual bool OnInit();
|
||||
virtual int OnExit();
|
||||
};
|
||||
|
||||
#endif // SLEEPYHEADAPP_H
|
512
SleepyHeadMain.cpp
Normal file
512
SleepyHeadMain.cpp
Normal file
@ -0,0 +1,512 @@
|
||||
/***************************************************************
|
||||
* 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 <wx/app.h>
|
||||
#include <wx/msgdlg.h>
|
||||
#include <wx/dirdlg.h>
|
||||
#include <wx/progdlg.h>
|
||||
|
||||
#include "SleepyHeadMain.h"
|
||||
#include "sleeplib/profiles.h"
|
||||
//#include "graphs/sleepflagsgraph.h"
|
||||
//#include "graphs/cpap_wavegraph.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)
|
||||
{
|
||||
loader_progress=new wxProgressDialog(wxT("SleepyHead"),wxT("Please Wait..."),100,this, wxPD_APP_MODAL|wxPD_AUTO_HIDE|wxPD_SMOOTH);
|
||||
loader_progress->Hide();
|
||||
wxString title=wxTheApp->GetAppName()+wxT(" v")+wxString(AutoVersion::FULLVERSION_STRING,wxConvUTF8);
|
||||
SetTitle(title);
|
||||
//wxDisableAsserts();
|
||||
// Create AUINotebook Tabs
|
||||
wxCommandEvent dummy;
|
||||
OnViewMenuSummary(dummy); // Summary Page
|
||||
OnViewMenuDaily(dummy); // Daily Page
|
||||
|
||||
#if wxUSE_STATUSBAR
|
||||
//statusBar->SetStatusText(_("Hello!"), 0);
|
||||
statusBar->SetStatusText(wxbuildinfo(long_f), 1);
|
||||
#endif
|
||||
}
|
||||
|
||||
SleepyHeadFrame::~SleepyHeadFrame()
|
||||
{
|
||||
if (loader_progress) {
|
||||
loader_progress->Hide();
|
||||
loader_progress->Destroy();
|
||||
delete loader_progress;
|
||||
}
|
||||
}
|
||||
|
||||
void SleepyHeadFrame::OnClose(wxCloseEvent &event)
|
||||
{
|
||||
Destroy();
|
||||
}
|
||||
|
||||
void SleepyHeadFrame::OnQuit(wxCommandEvent &event)
|
||||
{
|
||||
Destroy();
|
||||
}
|
||||
|
||||
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."); //,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) {
|
||||
wxString path=dd.GetPath();
|
||||
|
||||
Profile *p=Profiles::Get();
|
||||
|
||||
loader_progress->Update(0);
|
||||
loader_progress->Show();
|
||||
if (p) p->Import(path);
|
||||
loader_progress->Show(false);
|
||||
}
|
||||
int idx=main_auinotebook->GetPageIndex(daily);
|
||||
if (idx!=wxNOT_FOUND) {
|
||||
daily->RefreshData();
|
||||
}
|
||||
idx=main_auinotebook->GetPageIndex(summary);
|
||||
if (idx!=wxNOT_FOUND) {
|
||||
summary->RefreshData();
|
||||
}
|
||||
summary->Refresh();
|
||||
daily->Refresh();
|
||||
Refresh();
|
||||
}
|
||||
void SleepyHeadFrame::OnViewMenuDaily( wxCommandEvent& event )
|
||||
{
|
||||
int idx=main_auinotebook->GetPageIndex(daily);
|
||||
if (idx==wxNOT_FOUND) {
|
||||
daily=new Daily(this);
|
||||
main_auinotebook->AddPage(daily,_("Daily"),true);
|
||||
} else {
|
||||
main_auinotebook->SetSelection(idx);
|
||||
}
|
||||
}
|
||||
void SleepyHeadFrame::OnViewMenuSummary( wxCommandEvent& event )
|
||||
{
|
||||
int idx=main_auinotebook->GetPageIndex(summary);
|
||||
if (idx==wxNOT_FOUND) {
|
||||
summary=new Summary(this);
|
||||
main_auinotebook->AddPage(summary,_("Summary"),true);
|
||||
} else {
|
||||
main_auinotebook->SetSelection(idx);
|
||||
}
|
||||
}
|
||||
|
||||
Summary::Summary(wxWindow *win)
|
||||
:SummaryPanel(win)
|
||||
{
|
||||
machine=NULL;
|
||||
ahidata=new HistoryData(machine,30);
|
||||
AHI=new gGraphWindow(ScrolledWindow,-1,wxT("AHI"),wxPoint(0,0), wxSize(400,200), wxNO_BORDER);
|
||||
AHI->SetMargins(10,15,60,80);
|
||||
AHI->AddLayer(new gBarChart(ahidata,wxRED));
|
||||
fgSizer->Add(AHI,1,wxEXPAND);
|
||||
|
||||
pressure=new HistoryCodeData(machine,CPAP_PressureAverage,30);
|
||||
PRESSURE=new gGraphWindow(ScrolledWindow,-1,wxT("Average Pressure"),wxPoint(0,0), wxSize(400,200), wxNO_BORDER);
|
||||
PRESSURE->SetMargins(10,15,60,80);
|
||||
PRESSURE->AddLayer(new gBarChart(pressure,wxBLUE));
|
||||
fgSizer->Add(PRESSURE,1,wxEXPAND);
|
||||
|
||||
leak=new HistoryCodeData(machine,CPAP_LeakAverage,30);
|
||||
LEAK=new gGraphWindow(ScrolledWindow,-1,wxT("Average Leak"),wxPoint(0,0), wxSize(400,200), wxNO_BORDER);
|
||||
LEAK->SetMargins(10,15,60,80);
|
||||
LEAK->AddLayer(new gBarChart(leak,wxYELLOW));
|
||||
fgSizer->Add(LEAK,1,wxEXPAND);
|
||||
|
||||
RefreshData();
|
||||
|
||||
}
|
||||
Summary::~Summary()
|
||||
{
|
||||
}
|
||||
|
||||
void Summary::RefreshData()
|
||||
{
|
||||
if (!machine) {
|
||||
Profile *p=Profiles::Get();
|
||||
vector<Machine *>vm=p->GetMachines(MT_CPAP);
|
||||
if (vm.size()>=1) {
|
||||
machine=vm[0];
|
||||
} else machine=NULL;
|
||||
ahidata->SetMachine(machine);
|
||||
pressure->SetMachine(machine);
|
||||
leak->SetMachine(machine);
|
||||
}
|
||||
ahidata->Update();
|
||||
pressure->Update();
|
||||
leak->Update();
|
||||
wxString submodel=_("Unknown Model");
|
||||
double ahi=ahidata->GetAverage();
|
||||
double avp=pressure->GetAverage();
|
||||
|
||||
wxString html=wxT("<html><body leftmargin=0 rightmargin=0 topmargin=0 marginwidth=0 marginheight=0><table cellspacing=2 cellpadding=0>\n");
|
||||
|
||||
if (machine) {
|
||||
html=html+wxT("<tr><td colspan=2 align=center><i>")+_("Machine Information")+wxT("</i></td></tr>\n");
|
||||
if (machine->properties.find(wxT("SubModel"))!=machine->properties.end())
|
||||
submodel=wxT(" <br>\n ")+machine->properties[wxT("SubModel")];
|
||||
html=html+wxT("<tr><td colspan=2 align=center><b>")+machine->properties[wxT("Brand")]+wxT("</b> <br/>")+machine->properties[wxT("Model")]+wxT(" ")+machine->properties[wxT("ModelNumber")]+submodel+wxT("</td></tr>\n");
|
||||
html=html+wxT("<tr><td> </td><td> </td></tr>\n");
|
||||
html=html+wxT("<tr><td colspan=2 align=left><i>")+_("Indice Averages")+wxT("</i></td></tr>\n");
|
||||
html=html+wxT("<tr><td><b>")+_("AHI")+wxT("</b></td><td>")+wxString::Format(wxT("%0.2f"),ahi)+wxT("</td></tr>\n");
|
||||
html=html+wxT("<tr><td><b>")+_("Pressure")+wxT("</b></td><td>")+wxString::Format(wxT("%0.2fcmH2O"),avp)+wxT("</td></tr>\n");
|
||||
html=html+wxT("<tr><td><b>")+_("Mask Leaks")+wxT("</b></td><td>")+wxString::Format(wxT("%0.2f"),leak->GetAverage())+wxT("</td></tr>\n");
|
||||
html=html+wxT("</table>");
|
||||
} else {
|
||||
html=html+_("Please import some data.");
|
||||
}
|
||||
html+=wxT("</body></html>");
|
||||
HTMLInfo->SetPage(html);
|
||||
|
||||
|
||||
}
|
||||
|
||||
Daily::Daily(wxWindow *win)
|
||||
:DailyPanel(win)
|
||||
{
|
||||
Profile *p=Profiles::Get();
|
||||
vector<Machine *>vm=p->GetMachines(MT_CPAP);
|
||||
wxString s;
|
||||
if (vm.size()>=1) {
|
||||
machine=vm[0];
|
||||
} else machine=NULL;
|
||||
|
||||
TAP=new gGraphWindow(ScrolledWindow,-1,wxT("Time@Pressure"),wxPoint(0,0), wxSize(600,50), wxNO_BORDER);
|
||||
TAP->SetMargins(20,15,5,50);
|
||||
AddData(tap=new TAPData());
|
||||
TAP->AddLayer(new gCandleStick(tap));
|
||||
|
||||
G_AHI=new gGraphWindow(ScrolledWindow,-1,wxT("Event Breakdown"),wxPoint(0,0), wxSize(600,50), 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));
|
||||
LEAK=new gGraphWindow(ScrolledWindow,-1,wxT("Mask Leaks"),wxPoint(0,0), wxSize(600,150), wxNO_BORDER);
|
||||
LEAK->AddLayer(new gLineChart(leakdata,wxPURPLE,4096,false));
|
||||
|
||||
AddData(prd=new PressureData(CPAP_Pressure));
|
||||
PRD=new gGraphWindow(ScrolledWindow,-1,wxT("Pressure"),wxPoint(0,0), wxSize(600,150), wxNO_BORDER);
|
||||
PRD->AddLayer(new gLineChart(prd,wxDARK_GREEN,4096,false));
|
||||
|
||||
AddData(frw=new FlowData());
|
||||
FRW=new gGraphWindow(ScrolledWindow,-1,wxT("Flow Rate"),wxPoint(0,0), wxSize(600,150), wxNO_BORDER);
|
||||
// FRW->SetMargins(10,15,25,80);
|
||||
|
||||
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));
|
||||
|
||||
FRW->AddLayer(new gLineChart(frw,wxBLACK,200000,true));
|
||||
FRW->AddLayer(new gLineOverlayBar(flags[6],wxYELLOW,wxT("RE")));
|
||||
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 gLineOverlayBar(flags[0],wxGREEN2,wxT("CSR")));
|
||||
|
||||
SF=new gGraphWindow(ScrolledWindow,-1,wxT("Sleep Flags"),wxPoint(0,0), wxSize(600,150), wxNO_BORDER);
|
||||
SF->SetMargins(10,15,20,80);
|
||||
|
||||
SF->AddLayer(new gFlagsLine(flags[6],wxYELLOW,wxT("RE"),6,7));
|
||||
SF->AddLayer(new gFlagsLine(flags[5],wxRED,wxT("VS"),5,7));
|
||||
SF->AddLayer(new gFlagsLine(flags[4],wxBLACK,wxT("FL"),4,7));
|
||||
SF->AddLayer(new gFlagsLine(flags[3],wxBLUE,wxT("H"),3,7));
|
||||
SF->AddLayer(new gFlagsLine(flags[2],wxAQUA,wxT("OA"),2,7));
|
||||
SF->AddLayer(new gFlagsLine(flags[1],wxPURPLE,wxT("CA"),1,7));
|
||||
SF->AddLayer(new gFlagsLine(flags[0],wxGREEN2,wxT("CSR"),0,7));
|
||||
//l=new gBarChart(graphdata,wxHORIZONTAL);
|
||||
//graph->AddLayer(l);
|
||||
//graph->SetData(graphdata);
|
||||
/*l=new gBarChart(wxHORIZONTAL);
|
||||
graph2->AddLayer(l); */
|
||||
//m_mgr2.AddPane(graph,wxLEFT);
|
||||
//wxAuiPaneInfo &z=m_mgr2.GetPane(graph);
|
||||
//z.Caption(graph->GetTitle());
|
||||
//z.CloseButton(false);
|
||||
|
||||
|
||||
fgSizer->Add(SF,1,wxEXPAND);
|
||||
fgSizer->Add(G_AHI,1,wxEXPAND);
|
||||
fgSizer->Add(FRW,1,wxEXPAND);
|
||||
fgSizer->Add(PRD,1,wxEXPAND);
|
||||
fgSizer->Add(LEAK,1,wxEXPAND);
|
||||
fgSizer->Add(TAP,1,wxEXPAND);
|
||||
|
||||
|
||||
//m_mgr2.Update();
|
||||
//DailyGraphHolder->Add(graph,1,wxEXPAND);
|
||||
foobar_datehack=false;
|
||||
RefreshData();
|
||||
|
||||
}
|
||||
Daily::~Daily()
|
||||
{
|
||||
|
||||
}
|
||||
void Daily::RefreshData()
|
||||
{
|
||||
if (!machine) {
|
||||
Profile *p=Profiles::Get();
|
||||
vector<Machine *>vm=p->GetMachines(MT_CPAP);
|
||||
wxString s;
|
||||
if (vm.size()>=1) {
|
||||
machine=vm[0];
|
||||
} else machine=NULL;
|
||||
}
|
||||
|
||||
wxDateTime day=Calendar->GetDate();
|
||||
day.ResetTime();
|
||||
day.SetHour(0);
|
||||
//et-=wxTimeSpan::Days(1);
|
||||
UpdateGraphs(day);
|
||||
wxCalendarEvent ev;
|
||||
ev.SetDate(day);
|
||||
OnCalendarMonth(ev);
|
||||
OnCalendarDay(ev);
|
||||
}
|
||||
///usr/local/bin/upx ./bin/Windows/SleepyHead
|
||||
|
||||
void Daily::OnCalendarDay( wxCalendarEvent& event )
|
||||
{
|
||||
if (foobar_datehack) {
|
||||
OnCalendarMonth(event);
|
||||
foobar_datehack=false;
|
||||
}
|
||||
if (!machine) return;
|
||||
|
||||
wxDateTime day=event.GetDate();
|
||||
day.ResetTime();
|
||||
day.SetHour(0);
|
||||
day-=wxTimeSpan::Days(1);
|
||||
Day *d;
|
||||
if (machine && (machine->day.find(day)!=machine->day.end()) && (d=machine->day[day]) && (d->size()>0) && ((d->last()-d->first())>wxTimeSpan::Minutes(15))) {
|
||||
//HTMLInfo->SetPage(wxT(""));
|
||||
UpdateGraphs(day);
|
||||
|
||||
// Session *s=(*machine->day[day])[0];
|
||||
PRTypes pr=(PRTypes)d->summary_max(CPAP_PressureReliefType);
|
||||
wxString epr=PressureReliefNames[pr]+wxString::Format(wxT(" x%i"),(int)d->summary_max(CPAP_PressureReliefSetting));
|
||||
CPAPMode mode=(CPAPMode)d->summary_max(CPAP_Mode);
|
||||
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();
|
||||
wxString submodel=_("Unknown Model");
|
||||
|
||||
|
||||
wxString html=wxT("<html><body leftmargin=0 rightmargin=0 topmargin=0 marginwidth=0 marginheight=0><table cellspacing=2 cellpadding=0>\n");
|
||||
html=html+wxT("<tr><td colspan=2 align=center><i>")+_("Machine Information")+wxT("</i></td></tr>\n");
|
||||
if (machine->properties.find(wxT("SubModel"))!=machine->properties.end())
|
||||
submodel=wxT(" <br>")+machine->properties[wxT("SubModel")];
|
||||
html=html+wxT("<tr><td colspan=2 align=center><b>")+machine->properties[wxT("Brand")]+wxT("</b> <br>")+machine->properties[wxT("Model")]+wxT(" ")+machine->properties[wxT("ModelNumber")]+submodel+wxT("</td></tr>\n");
|
||||
html=html+wxT("<tr><td> </td><td> </td></tr>\n");
|
||||
html=html+wxT("<tr><td colspan=2 align=center><i>")+_("Sleep Times")+wxT("</i></td></tr>\n");
|
||||
html=html+wxT("<tr><td><b>")+_("Date")+wxT("</b></td><td>")+machine->day[day]->first().Format(wxT("%x"))+wxT("</td></tr>\n");
|
||||
html=html+wxT("<tr><td><b>")+_("Sleep")+wxT("</b></td><td>")+machine->day[day]->first().Format(wxT("%I:%M%p"))+wxT("</td></tr>\n");
|
||||
html=html+wxT("<tr><td><b>")+_("Wake")+wxT("</b></td><td>")+machine->day[day]->last().Format(wxT("%I:%M%p"))+wxT("</td></tr>\n");
|
||||
html=html+wxT("<tr><td> </td><td> </td></tr>\n");
|
||||
html=html+wxT("<tr><td colspan=2 align=center><i>")+_("Indices")+wxT("</i></td></tr>\n");
|
||||
html=html+wxT("<tr><td><b>")+_("AHI")+wxT("</b></td><td>")+wxString::Format(wxT("%0.2f"),ahi)+wxT("</td></tr>\n");
|
||||
html=html+wxT("<tr><td><b>")+_("Obstructive")+wxT("</b></td><td>")+wxString::Format(wxT("%0.2f"),oai)+wxT("</td></tr>\n");
|
||||
html=html+wxT("<tr><td><b>")+_("Hypopnea")+wxT("</b></td><td>")+wxString::Format(wxT("%0.2f"),hi)+wxT("</td></tr>\n");
|
||||
html=html+wxT("<tr><td><b>")+_("ClearAirway")+wxT("</b></td><td>")+wxString::Format(wxT("%0.2f"),cai)+wxT("</td></tr>\n");
|
||||
html=html+wxT("<tr><td><b>")+_("RERA")+wxT("</b></td><td>")+wxString::Format(wxT("%0.2f"),rei)+wxT("</td></tr>\n");
|
||||
html=html+wxT("<tr><td><b>")+_("FlowLimit")+wxT("</b></td><td>")+wxString::Format(wxT("%0.2f"),fli)+wxT("</td></tr>\n");
|
||||
html=html+wxT("<tr><td><b>")+_("Vsnore")+wxT("</b></td><td>")+wxString::Format(wxT("%0.2f"),vsi)+wxT("</td></tr>\n");
|
||||
html=html+wxT("<tr><td><b>")+_("CSR")+wxT("</b></td><td>")+wxString::Format(wxT("%0.2f%%"),csr)+wxT("</td></tr>\n");
|
||||
html=html+wxT("<tr><td> </td><td> </td></tr>\n");
|
||||
html=html+wxT("<tr><td colspan=2 align=center><i>")+_("Other Information")+wxT("</i></td></tr>\n");
|
||||
|
||||
|
||||
// html=html+wxT("<tr><td> </td><td> </td></tr>\n");
|
||||
// html=html+wxT("<tr><td colspan=2 align=center><i>Session Informaton</i></td></tr>\n");
|
||||
html=html+wxT("<tr><td><b>")+_("Mode")+wxT("</b></td><td>")+modestr+wxT("</td></tr>\n");
|
||||
html=html+wxT("<tr><td><b>")+_("Relief")+wxT("</b></td><td>")+epr+wxT("</td></tr>\n");
|
||||
html=html+wxT("<tr><td><b>")+_("Avg Leak")+wxT("</b></td><td>")+wxString::Format(wxT("%.2f"),d->summary_avg(CPAP_LeakAverage))+wxT("</td></tr>\n");
|
||||
if (mode==MODE_CPAP) {
|
||||
html=html+wxT("<tr><td><b>")+_("Pressure")+wxT("</b></td><td>")+wxString::Format(wxT("%.1fcmH2O"),d->summary_min(CPAP_PressureMin))+wxT("</td></tr>\n");
|
||||
} else if (mode==MODE_APAP) {
|
||||
html=html+wxT("<tr><td><b>")+_("Pressure-Min")+wxT("</b></td><td>")+wxString::Format(wxT("%.1fcmH2O"),d->summary_min(CPAP_PressureMinAchieved))+wxT("</td></tr>\n");
|
||||
html=html+wxT("<tr><td><b>")+_("Pressure-Max")+wxT("</b></td><td>")+wxString::Format(wxT("%.1fcmH2O"),d->summary_max(CPAP_PressureMax))+wxT("</td></tr>\n");
|
||||
html=html+wxT("<tr><td><b>")+_("Avg Pressure")+wxT("</b></td><td>")+wxString::Format(wxT("%.1fcmH2O"),d->summary_avg(CPAP_PressureAverage))+wxT("</td></tr>\n");
|
||||
html=html+wxT("<tr><td><b>")+_("90% Pressure")+wxT("</b></td><td>")+wxString::Format(wxT("%.1fcmH2O"),d->summary_avg(CPAP_PressurePercentValue))+wxT("</td></tr>\n");
|
||||
}
|
||||
|
||||
html=html+wxT("<tr><td> </td><td> </td></tr>\n");
|
||||
html=html+wxT("<tr><td colspan=2 align=center><i>")+_("System Settings")+wxT("</i></td></tr>\n");
|
||||
|
||||
|
||||
html=html+wxT("<tr><td><b>")+_("Ramp-Time")+wxT("</b></td><td>")+wxString::Format(wxT("%imin"),(int)d->summary_max(CPAP_RampTime))+wxT("</td></tr>\n");
|
||||
html=html+wxT("<tr><td><b>")+_("Ramp-Prs.")+wxT("</b></td><td>")+wxString::Format(wxT("%.1fcmH2O"),d->summary_min(CPAP_RampStartingPressure))+wxT("</td></tr>\n");
|
||||
// check HumidiferStatus..
|
||||
wxString humid;
|
||||
if (bool(d->summary_max(CPAP_HumidifierStatus))) {
|
||||
humid=wxString::Format(wxT("x%i"),(int)d->summary_max(CPAP_HumidifierSetting));
|
||||
} else humid=wxT("No");
|
||||
html=html+wxT("<tr><td><b>")+_("Humidifier")+wxT("</b></td><td>")+humid+wxT("</td></tr>\n");
|
||||
html=html+wxT("<tr><td><b>")+_("System-Lock")+wxT("</b></td><td>")+(bool(d->summary_max(PRS1_SystemLockStatus)) ? _("On") : _("Off"))+wxT("</td></tr>\n");
|
||||
html=html+wxT("<tr><td><b>")+_("Auto-Off")+wxT("</b></td><td>")+(bool(d->summary_max(PRS1_AutoOff)) ? _("On") : _("Off"))+wxT("</td></tr>\n");
|
||||
html=html+wxT("<tr><td><b>")+_("Mask-Alert")+wxT("</b></td><td>")+(bool(d->summary_max(PRS1_MaskAlert)) ? _("On") : _("Off"))+wxT("</td></tr>\n");
|
||||
html=html+wxT("<tr><td><b>")+_("Show-AHI")+wxT("</b></td><td>")+(bool(d->summary_max(PRS1_ShowAHI)) ? _("On") : _("Off"))+wxT("</td></tr>\n");
|
||||
html=html+wxT("<tr><td><b>")+_("Hose-Size")+wxT("</b></td><td>")+(bool(d->summary_max(PRS1_HoseDiameter)) ? _("22mm") : _("15mm"))+wxT("</td></tr>\n");
|
||||
html=html+wxT("<tr><td><b>")+_("Sys-Resist.")+wxT("</b></td><td>")+wxString::Format(wxT("%i"),int(d->summary_max(PRS1_SystemResistanceStatus)))+wxT("</td></tr>\n");
|
||||
//PRS1_SystemLockStatus
|
||||
|
||||
html=html+wxT("</table>");
|
||||
/* for (auto i=s->summary.begin();i!=s->summary.end();i++) {
|
||||
MachineCode c=(*i).first;
|
||||
wxString name;
|
||||
if (DefaultMCShortNames.find(c)!=DefaultMCShortNames.end()) name=DefaultMCShortNames[c];
|
||||
else name=wxString::Format(wxT("%04i"),(int)c);
|
||||
html+=name+wxT(" = ")+(*i).second.GetString()+wxT("<br/>\n");
|
||||
} */
|
||||
html+=wxT("</body></html>");
|
||||
HTMLInfo->SetPage(html);
|
||||
} else {
|
||||
HTMLInfo->SetPage(_("No CPAP Machine Data Available"));
|
||||
UpdateGraphs(wxInvalidDateTime);
|
||||
}
|
||||
|
||||
}
|
||||
void Daily::UpdateGraphs(wxDateTime d)
|
||||
{
|
||||
Day *day=NULL;
|
||||
if (!machine) return;
|
||||
if (d!=wxInvalidDateTime) {
|
||||
if (machine->day.find(d)!=machine->day.end()) {
|
||||
day=machine->day[d];
|
||||
}
|
||||
}
|
||||
for (auto g=Data.begin();g!=Data.end();g++) {
|
||||
|
||||
(*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 ((machine->day.find(d)!=machine->day.end()) && ((machine->day[d]->last() - machine->day[d]->first())>wxTimeSpan::Minutes(15))) {
|
||||
#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);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
|
79
SleepyHeadMain.h
Normal file
79
SleepyHeadMain.h
Normal file
@ -0,0 +1,79 @@
|
||||
/***************************************************************
|
||||
* Name: SleepyHeadMain.h
|
||||
* Purpose: Defines Application Frame
|
||||
* Author: Mark Watkins (jedimark64@users.sourceforge.net)
|
||||
* Created: 2011-05-20
|
||||
* Copyright: Mark Watkins (http://sourceforge.net/projects/sleepyhead/)
|
||||
* License: GPL
|
||||
**************************************************************/
|
||||
|
||||
#ifndef SLEEPYHEADMAIN_H
|
||||
#define SLEEPYHEADMAIN_H
|
||||
|
||||
|
||||
|
||||
#include "SleepyHeadApp.h"
|
||||
|
||||
#include "GUIFrame.h"
|
||||
#include "sleeplib/machine.h"
|
||||
#include "graphs/graph.h"
|
||||
//#include "graphs/sleepflagsgraph.h"
|
||||
//#include "graphs/cpap_wavegraph.h"
|
||||
//#include "graphs/cpap_pressure.h"
|
||||
|
||||
class Summary:public SummaryPanel
|
||||
{
|
||||
public:
|
||||
Summary(wxWindow *win);
|
||||
virtual ~Summary();
|
||||
Machine *machine;
|
||||
HistoryData *ahidata,*pressure,*leak;
|
||||
gGraphWindow *AHI,*PRESSURE,*LEAK;
|
||||
void RefreshData();
|
||||
|
||||
};
|
||||
|
||||
class Daily:public DailyPanel
|
||||
{
|
||||
public:
|
||||
Daily(wxWindow *win);
|
||||
virtual ~Daily();
|
||||
void RefreshData();
|
||||
|
||||
Machine *machine;
|
||||
protected:
|
||||
virtual void OnCalendarDay( wxCalendarEvent& event );
|
||||
virtual void OnCalendarMonth( wxCalendarEvent& event );
|
||||
void AddData(gPointData *d) { Data.push_back(d); };
|
||||
void UpdateGraphs(wxDateTime date);
|
||||
|
||||
bool foobar_datehack;
|
||||
gPointData *tap,*g_ahi,*frw,*prd,*leakdata;
|
||||
gPointData *flags[8];
|
||||
gGraphWindow *PRD,*FRW,*G_AHI,*TAP,*LEAK,*SF;
|
||||
|
||||
|
||||
list<gPointData *> Data;
|
||||
};
|
||||
|
||||
class SleepyHeadFrame: public GUIFrame
|
||||
{
|
||||
public:
|
||||
SleepyHeadFrame(wxFrame *frame);
|
||||
~SleepyHeadFrame();
|
||||
private:
|
||||
virtual void OnClose(wxCloseEvent& event);
|
||||
virtual void OnQuit(wxCommandEvent& event);
|
||||
virtual void OnAbout(wxCommandEvent& event);
|
||||
virtual void OnImportSD(wxCommandEvent& event);
|
||||
virtual void OnViewMenuDaily(wxCommandEvent& event);
|
||||
virtual void OnViewMenuSummary(wxCommandEvent& event);
|
||||
|
||||
Summary *summary;
|
||||
Daily *daily;
|
||||
Machine *machine;
|
||||
Profile *profile;
|
||||
|
||||
};
|
||||
|
||||
#endif // SLEEPYHEADMAIN_H
|
1097
WxWizFrame.fbp
Normal file
1097
WxWizFrame.fbp
Normal file
File diff suppressed because it is too large
Load Diff
1868
graphs/graph.cpp
Normal file
1868
graphs/graph.cpp
Normal file
File diff suppressed because it is too large
Load Diff
452
graphs/graph.h
Normal file
452
graphs/graph.h
Normal file
@ -0,0 +1,452 @@
|
||||
/*
|
||||
gGraph Graphing System Header File
|
||||
|
||||
Author: Mark Watkins <jedimark64@users.sourceforge.net>
|
||||
License: LGPL
|
||||
*/
|
||||
#ifndef GRAPH_H
|
||||
#define GRAPH_H
|
||||
|
||||
#include <sleeplib/machine.h>
|
||||
#include <list>
|
||||
|
||||
|
||||
#if !wxCHECK_VERSION(2,9,0)
|
||||
extern wxColor *wxYELLOW;
|
||||
#endif
|
||||
extern wxColor *wxAQUA;
|
||||
extern wxColor *wxPURPLE;
|
||||
extern wxColor *wxGREEN2;
|
||||
extern wxColor *wxLIGHT_YELLOW;
|
||||
extern wxColor *wxDARK_GREEN;
|
||||
|
||||
#define MIN(a,b) (a<b) ? a : b;
|
||||
#define MAX(a,b) (a<b) ? b : a;
|
||||
|
||||
class Point3D
|
||||
{
|
||||
public:
|
||||
Point3D() {x=0; y=0; z=0;};
|
||||
Point3D(double _x,double _y, double _z) { x=_x; y=_y; z=_z; };
|
||||
|
||||
double x,y,z;
|
||||
};
|
||||
|
||||
enum gDataType { gDT_Point, gDT_Point3D, gDT_Stacked, gDT_Segmented };
|
||||
|
||||
class gLayer;
|
||||
|
||||
class gGraphData
|
||||
{
|
||||
public:
|
||||
gGraphData(int mp,gDataType t=gDT_Point);
|
||||
virtual ~gGraphData();
|
||||
|
||||
virtual void Reload(Day *day=NULL) {};
|
||||
|
||||
virtual void Update(Day *day=NULL);
|
||||
//inline wxRealPoint & operator [](int i) { return vpoint[seg][i]; };
|
||||
//inline vector<double> & Vec(int i) { return yaxis[i]; };
|
||||
|
||||
//virtual inline const int & NP(int i) { return vnp[i]; };
|
||||
//virtual inline const int & MP(int i) { return vsize[i]; };
|
||||
inline const gDataType & Type() { return type; };
|
||||
|
||||
virtual inline double MaxX() { return max_x; };
|
||||
virtual inline double MinX() { return min_x; };
|
||||
virtual inline double MaxY() { return max_y; };
|
||||
virtual inline double MinY() { return min_y; };
|
||||
virtual inline void SetMaxX(double v) { max_x=v; if (max_x>real_max_x) max_x=real_max_x; };
|
||||
virtual inline void SetMinX(double v) { min_x=v; if (min_x<real_min_x) min_x=real_min_x; };
|
||||
virtual inline void SetMaxY(double v) { max_y=v; if (max_y>real_max_y) max_y=real_max_y; };
|
||||
virtual inline void SetMinY(double v) { min_y=v; if (min_y<real_min_y) min_y=real_min_y; };
|
||||
virtual inline double RealMaxX() { return real_max_x; };
|
||||
virtual inline double RealMinX() { return real_min_x; };
|
||||
virtual inline double RealMaxY() { return real_max_y; };
|
||||
virtual inline double RealMinY() { return real_min_y; };
|
||||
|
||||
inline void ResetX() { min_x=real_min_x; max_x=real_max_x; };
|
||||
inline void ResetY() { min_y=real_min_y; max_y=real_max_y; };
|
||||
|
||||
virtual inline int VC() { return vc; };
|
||||
|
||||
vector<int> np;
|
||||
vector<int> maxsize;
|
||||
const bool & IsReady() { return m_ready; };
|
||||
|
||||
void AddLayer(gLayer *g) { notify_layers.push_back(g); };
|
||||
protected:
|
||||
virtual void AddSegment(int max_points) {};
|
||||
|
||||
double real_min_x, real_max_x, real_min_y, real_max_y;
|
||||
double min_x, max_x, min_y, max_y;
|
||||
|
||||
int vc;
|
||||
gDataType type;
|
||||
int max_points;
|
||||
bool m_ready;
|
||||
|
||||
list<gLayer *> notify_layers;
|
||||
|
||||
};
|
||||
|
||||
class gPoint3DData:public gGraphData
|
||||
{
|
||||
public:
|
||||
gPoint3DData(int mp);
|
||||
virtual ~gPoint3DData();
|
||||
virtual void Reload(Day *day=NULL) {};
|
||||
virtual void AddSegment(int max_points);
|
||||
vector<Point3D *> point;
|
||||
};
|
||||
|
||||
class gPointData:public gGraphData
|
||||
{
|
||||
public:
|
||||
gPointData(int mp);
|
||||
virtual ~gPointData();
|
||||
virtual void Reload(Day *day=NULL){};
|
||||
virtual void AddSegment(int max_points);
|
||||
vector<wxRealPoint *> point;
|
||||
};
|
||||
|
||||
|
||||
|
||||
class gGraphWindow:public wxWindow // rename to gGraphWindow
|
||||
{
|
||||
public:
|
||||
gGraphWindow() {};
|
||||
gGraphWindow(wxWindow *parent, wxWindowID id,const wxString & title=wxT("Graph"),const wxPoint &pos = wxDefaultPosition,const wxSize &size = wxDefaultSize,long flags = 0);
|
||||
|
||||
virtual ~gGraphWindow();
|
||||
virtual void OnPaint(wxPaintEvent & event);
|
||||
virtual void OnSize(wxSizeEvent & event);
|
||||
//virtual void OnMouseWheel(wxMouseEvent &event);
|
||||
virtual void OnMouseMove(wxMouseEvent &event);
|
||||
virtual void OnMouseLeftDown(wxMouseEvent &event);
|
||||
virtual void OnMouseLeftRelease (wxMouseEvent &event);
|
||||
virtual void OnMouseRightDown(wxMouseEvent &event);
|
||||
virtual void OnMouseRightRelease(wxMouseEvent &event);
|
||||
|
||||
int GetScrX(void) const { return m_scrX; };
|
||||
int GetScrY(void) const { return m_scrY; };
|
||||
|
||||
void SetMargins(int top, int right, int bottom, int left);
|
||||
const wxString & Title(void ) { return m_title; };
|
||||
int GetTopMargin(void) const { return m_marginTop; };
|
||||
int GetBottomMargin(void) const { return m_marginBottom; };
|
||||
int GetLeftMargin(void) const { return m_marginLeft; };
|
||||
int GetRightMargin(void) const { return m_marginRight; };
|
||||
|
||||
inline int Width() { return m_scrX-m_marginLeft-m_marginRight; };
|
||||
inline int Height() { return m_scrY-m_marginTop-m_marginBottom; };
|
||||
|
||||
virtual double MinX();
|
||||
virtual double MaxX();
|
||||
virtual double MinY();
|
||||
virtual double MaxY();
|
||||
|
||||
virtual double RealMinX();
|
||||
virtual double RealMaxX();
|
||||
virtual double RealMinY();
|
||||
virtual double RealMaxY();
|
||||
|
||||
virtual void SetMinX(double v);
|
||||
virtual void SetMaxX(double v);
|
||||
virtual void SetMinY(double v);
|
||||
virtual void SetMaxY(double v);
|
||||
|
||||
virtual void SetXBounds(double minx, double maxx);
|
||||
virtual void ZoomX(double mult,int origin_px);
|
||||
virtual void ZoomXPixels(int x1, int x2); // Zoom between two selected points on screen
|
||||
virtual void MoveX(int i); // Move x bounds by i Pixels
|
||||
|
||||
inline int x2p(double x) {
|
||||
double xx=max_x-min_x;
|
||||
double w=(Width()/xx)*(x-min_x);
|
||||
return w+GetLeftMargin();
|
||||
};
|
||||
inline double p2x(int px) {
|
||||
double xx=max_x-min_x;
|
||||
double wx=px-GetLeftMargin();
|
||||
double ww=wx/Width();
|
||||
return min_x+(xx*ww);
|
||||
};
|
||||
inline int y2p(double y) {
|
||||
double yy=max_y-min_y;
|
||||
double h=(Height()/yy)*(y-min_y);
|
||||
return h+GetTopMargin();
|
||||
};
|
||||
inline double p2y(int py) {
|
||||
double yy=max_y-min_y;
|
||||
double hy=py-GetTopMargin();
|
||||
double hh=hy/Height();
|
||||
return min_y+(yy*hh);
|
||||
};
|
||||
|
||||
virtual void Update();
|
||||
//virtual void Update();
|
||||
void AddLayer(gLayer *l);
|
||||
|
||||
void DataChanged(gLayer *layer);
|
||||
|
||||
double max_x,min_x,max_y,min_y;
|
||||
double rmax_x,rmin_x,rmax_y,rmin_y;
|
||||
|
||||
protected:
|
||||
|
||||
std::list<gLayer *> layers;
|
||||
wxColour m_bgColour; //!< Background Colour
|
||||
wxColour m_fgColour; //!< Foreground Colour
|
||||
wxColour m_axColour; //!< Axes Colour
|
||||
wxString m_title;
|
||||
int m_scrX; //!< Current view's X dimension
|
||||
int m_scrY; //!< Current view's Y dimension
|
||||
wxPoint m_mouseLClick,m_mouseRClick,m_mouseRClick_start;
|
||||
|
||||
int m_marginTop, m_marginRight, m_marginBottom, m_marginLeft;
|
||||
|
||||
wxRect m_mouseRBrect,m_mouseRBlast;
|
||||
bool m_mouseLDown,m_mouseRDown,m_datarefresh;
|
||||
|
||||
DECLARE_DYNAMIC_CLASS(gGraphWindow)
|
||||
DECLARE_EVENT_TABLE()
|
||||
};
|
||||
|
||||
//Borrows a lot of concepts from wxmathplot
|
||||
class gLayer
|
||||
{
|
||||
public:
|
||||
gLayer(gPointData *g=NULL,wxString title=wxT(""));
|
||||
virtual ~gLayer();
|
||||
//virtual void Update() { data=gd; };
|
||||
virtual void Plot(wxDC & dc, gGraphWindow & w);
|
||||
vector<const wxColor *> color;
|
||||
virtual void SetData(gPointData * gd) { data=gd; };
|
||||
virtual gPointData * GetData() { return data; };
|
||||
|
||||
virtual void DataChanged(gGraphData *src) {
|
||||
for (auto i=m_graph.begin();i!=m_graph.end();i++)
|
||||
(*i)->DataChanged(this);
|
||||
|
||||
}; // Notify signal sent from gGraphData.. pass on to the graph so it can que a refresh and update stuff.
|
||||
|
||||
virtual double MinX() { if (data) return data->MinX(); return 0;};
|
||||
virtual double MaxX() { if (data) return data->MaxX(); return 0;};
|
||||
virtual double MinY() { if (data) return data->MinY(); return 0;};
|
||||
virtual double MaxY() { if (data) return data->MaxY(); return 0;};
|
||||
|
||||
virtual double RealMinX() { if (data) return data->RealMinX(); return 0;};
|
||||
virtual double RealMaxX() { if (data) return data->RealMaxX(); return 0;};
|
||||
virtual double RealMinY() { if (data) return data->RealMinY(); return 0;};
|
||||
virtual double RealMaxY() { if (data) return data->RealMaxY(); return 0;};
|
||||
|
||||
virtual void SetMinX(double v) { if (data) data->SetMinX(v); };
|
||||
virtual void SetMaxX(double v) { if (data) data->SetMaxX(v); };
|
||||
virtual void SetMinY(double v) { if (data) data->SetMinY(v); };
|
||||
virtual void SetMaxY(double v) { if (data) data->SetMaxY(v); };
|
||||
|
||||
virtual inline int x2p(gGraphWindow & g,double x) {
|
||||
double xx=data->MaxX()-data->MinX();
|
||||
double w=(g.Width()/xx)*(x-data->MinX());
|
||||
return w+g.GetLeftMargin();
|
||||
};
|
||||
virtual inline double p2x(gGraphWindow & g,int px) {
|
||||
double xx=data->MaxX()-data->MinX();
|
||||
double wx=px-g.GetLeftMargin();
|
||||
double ww=wx/g.Width();
|
||||
return data->MinX()+(xx*ww);
|
||||
};
|
||||
virtual inline int y2p(gGraphWindow & g,double y) {
|
||||
double yy=data->MaxY()-data->MinY();
|
||||
double h=(g.Height()/yy)*(y-data->MinY());
|
||||
return h+g.GetTopMargin();
|
||||
};
|
||||
virtual inline double p2y(gGraphWindow & g,int py) {
|
||||
double yy=data->MaxY()-data->MinY();
|
||||
double hy=py-g.GetTopMargin();
|
||||
double hh=hy/g.Height();
|
||||
return data->MinY()+(yy*hh);
|
||||
};
|
||||
void NotifyGraphWindow(gGraphWindow *g) { m_graph.push_back(g); };
|
||||
|
||||
protected:
|
||||
bool m_visible;
|
||||
bool m_movable;
|
||||
wxString m_title;
|
||||
gPointData *data; // Data source
|
||||
list<gGraphWindow *> m_graph; // notify list of graphs that attach this layer.
|
||||
};
|
||||
|
||||
class gCandleStick:public gLayer
|
||||
{
|
||||
public:
|
||||
gCandleStick(gPointData *d=NULL,wxOrientation o=wxHORIZONTAL);
|
||||
virtual ~gCandleStick();
|
||||
|
||||
virtual void Plot(wxDC & dc, gGraphWindow & w);
|
||||
void AddName(wxString name) { m_names.push_back(name); };
|
||||
|
||||
protected:
|
||||
wxOrientation m_direction;
|
||||
vector<wxString> m_names;
|
||||
|
||||
};
|
||||
|
||||
class gLineChart:public gLayer
|
||||
{
|
||||
public:
|
||||
gLineChart(gPointData *d=NULL,const wxColor * col=wxBLACK,int dlsize=4096,bool a=true);
|
||||
virtual ~gLineChart();
|
||||
|
||||
virtual void Plot(wxDC & dc, gGraphWindow & w);
|
||||
virtual const wxString & FormatX(double v) { static wxString t; wxDateTime d; d.Set(v); t=d.Format(wxT("%H:%M")); return t; };
|
||||
//virtual const wxString & FormatX(double v) { static wxString t; t=wxString::Format(wxT("%.1f"),v); return t; };
|
||||
virtual const wxString & FormatY(double v) { static wxString t; t=wxString::Format(wxT("%.1f"),v); return t; };
|
||||
|
||||
virtual void DrawXTicks(wxDC & dc,gGraphWindow &w);
|
||||
virtual void DrawYTicks(wxDC & dc,gGraphWindow &w);
|
||||
|
||||
protected:
|
||||
float m_yminor_ticks;
|
||||
float m_ymajor_ticks;
|
||||
bool m_accelerate;
|
||||
int m_drawlist_size;
|
||||
wxPoint *m_drawlist;
|
||||
bool m_show_grid;
|
||||
bool m_show_minor_grid;
|
||||
|
||||
};
|
||||
|
||||
class gLineOverlayBar:public gLayer
|
||||
{
|
||||
public:
|
||||
gLineOverlayBar(gPointData *d=NULL,const wxColor * col=wxBLACK,wxString _label=wxT(""));
|
||||
virtual ~gLineOverlayBar();
|
||||
|
||||
virtual void Plot(wxDC & dc, gGraphWindow & w);
|
||||
|
||||
protected:
|
||||
wxString label;
|
||||
};
|
||||
|
||||
class gFlagsLine:public gLayer
|
||||
{
|
||||
public:
|
||||
gFlagsLine(gPointData *d=NULL,const wxColor * col=wxBLACK,wxString _label=wxT(""),int _line_num=0,int _total_lines=0);
|
||||
virtual ~gFlagsLine();
|
||||
|
||||
virtual void Plot(wxDC & dc, gGraphWindow & w);
|
||||
|
||||
protected:
|
||||
wxString label;
|
||||
int line_num,total_lines;
|
||||
};
|
||||
|
||||
|
||||
class gBarChart:public gLayer
|
||||
{
|
||||
public:
|
||||
gBarChart(gPointData *d=NULL,const wxColor *col=NULL,wxOrientation o=wxHORIZONTAL);
|
||||
virtual ~gBarChart();
|
||||
|
||||
virtual void Plot(wxDC & dc, gGraphWindow & w);
|
||||
|
||||
protected:
|
||||
void DrawYTicks(wxDC & dc,gGraphWindow &w);
|
||||
|
||||
wxOrientation m_direction;
|
||||
float m_yminor_ticks;
|
||||
float m_ymajor_ticks;
|
||||
|
||||
// d.Set(i+2400000.5+.000001); // JDN vs MJD vs Rounding errors
|
||||
|
||||
virtual const wxString & FormatX(double v) { static wxString t; wxDateTime d; d.Set(v+2400000.5); t=d.Format(wxT("%e %b")); return t; };
|
||||
//virtual const wxString & FormatX(double v) { static wxString t; wxDateTime d; d.Set(vi+2400000.5); t=d.Format(wxT("%H:%M")); return t; };
|
||||
//virtual const wxString & FormatX(double v) { static wxString t; t=wxString::Format(wxT("%.1f"),v); return t; };
|
||||
virtual const wxString & FormatY(double v) { static wxString t; t=wxString::Format(wxT("%.1f"),v); return t; };
|
||||
|
||||
bool m_show_grid;
|
||||
bool m_show_minor_grid;
|
||||
|
||||
};
|
||||
|
||||
class FlagData:public gPointData
|
||||
{
|
||||
public:
|
||||
FlagData(MachineCode _code,double _value=0,int _field=-1,int _offset=-1);
|
||||
virtual ~FlagData();
|
||||
virtual void Reload(Day *day=NULL);
|
||||
protected:
|
||||
MachineCode code;
|
||||
double value;
|
||||
int field;
|
||||
int offset;
|
||||
};
|
||||
|
||||
class TAPData:public gPointData
|
||||
{
|
||||
public:
|
||||
TAPData();
|
||||
virtual ~TAPData();
|
||||
virtual void Reload(Day *day=NULL);
|
||||
};
|
||||
|
||||
class FlowData:public gPointData
|
||||
{
|
||||
public:
|
||||
FlowData();
|
||||
virtual ~FlowData();
|
||||
virtual void Reload(Day *day=NULL);
|
||||
};
|
||||
|
||||
class PressureData:public gPointData
|
||||
{
|
||||
public:
|
||||
PressureData(MachineCode _code,int _field=0);
|
||||
virtual ~PressureData();
|
||||
virtual void Reload(Day *day=NULL);
|
||||
protected:
|
||||
MachineCode code;
|
||||
int field;
|
||||
};
|
||||
|
||||
class AHIData:public gPointData
|
||||
{
|
||||
public:
|
||||
AHIData();
|
||||
virtual ~AHIData();
|
||||
virtual void Reload(Day *day=NULL);
|
||||
};
|
||||
|
||||
class HistoryData:public gPointData
|
||||
{
|
||||
public:
|
||||
HistoryData(Machine *_machine,int _days);
|
||||
virtual ~HistoryData();
|
||||
|
||||
void SetMachine(Machine *_machine) { machine=_machine; Reload(); };
|
||||
Machine * GetMachine() { return machine; };
|
||||
void SetDays(int i) { days=i; };
|
||||
int GetDays() { return days; };
|
||||
double GetAverage();
|
||||
|
||||
virtual double Calc(Day *day);
|
||||
virtual void Reload(Day *day=NULL);
|
||||
// virtual void Reload(Machine *machine=NULL);
|
||||
protected:
|
||||
Machine *machine;
|
||||
int days;
|
||||
};
|
||||
|
||||
class HistoryCodeData:public HistoryData
|
||||
{
|
||||
public:
|
||||
HistoryCodeData(Machine *_machine,MachineCode _code,int _days);
|
||||
virtual ~HistoryCodeData();
|
||||
virtual double Calc(Day *day);
|
||||
protected:
|
||||
MachineCode code;
|
||||
};
|
||||
|
||||
#endif // GRAPH_H
|
316
libs/sleeplib/binary_file.cpp
Normal file
316
libs/sleeplib/binary_file.cpp
Normal file
@ -0,0 +1,316 @@
|
||||
/*
|
||||
|
||||
SleepLib BinaryFile Implementation
|
||||
|
||||
Author: Mark Watkins <jedimark64@users.sourceforge.net>
|
||||
License: GPL
|
||||
|
||||
*/
|
||||
|
||||
#include <wx/filename.h>
|
||||
#include "binary_file.h"
|
||||
|
||||
BinaryFile::BinaryFile()
|
||||
{
|
||||
size=pos=0;
|
||||
buffer=NULL;
|
||||
}
|
||||
void BinaryFile::Close()
|
||||
{
|
||||
if (bf_mode==BF_WRITE) {
|
||||
//f.Write(buffer,pos);
|
||||
}
|
||||
f.Close();
|
||||
if (buffer) delete [] buffer;
|
||||
size=pos=0;
|
||||
}
|
||||
BinaryFile::~BinaryFile()
|
||||
{
|
||||
Close();
|
||||
}
|
||||
bool BinaryFile::Open(wxString filename,BFOpenMode mode)
|
||||
{
|
||||
pos=size=0;
|
||||
bf_mode=mode;
|
||||
if (wxFileExists(filename)) {
|
||||
if (mode==BF_WRITE) {
|
||||
// hmm.. file exists. unsure of best course of action here..
|
||||
return false;
|
||||
}
|
||||
size=wxFileName::GetSize(filename).GetLo();
|
||||
}
|
||||
wxString om;
|
||||
if (mode==BF_READ) om=wxT("rb");
|
||||
else if (mode==BF_WRITE) om=wxT("wb");
|
||||
|
||||
if (!f.Open(filename,om)) return false;
|
||||
|
||||
if (mode==BF_READ) {
|
||||
buffer=new char [size];
|
||||
size=f.Read(buffer,size);
|
||||
}
|
||||
return true;
|
||||
}
|
||||
bool BinaryFile::Unpack(wxInt8 & data)
|
||||
{
|
||||
if (pos>=size) return false;
|
||||
data=((wxInt8 *)buffer)[pos++];
|
||||
return true;
|
||||
}
|
||||
bool BinaryFile::Unpack(bool & data)
|
||||
{
|
||||
if (pos>=size) return false;
|
||||
data=((wxInt8 *)buffer)[pos++];
|
||||
return true;
|
||||
}
|
||||
|
||||
bool BinaryFile::Unpack(wxInt16 & data)
|
||||
{
|
||||
if (pos+1>=size) return false;
|
||||
data=((wxUint8 *)buffer)[pos++];
|
||||
data|=((wxInt8 *)buffer)[pos++] << 8;
|
||||
return true;
|
||||
}
|
||||
bool BinaryFile::Unpack(wxInt32 & data)
|
||||
{
|
||||
if (pos+3>=size) return false;
|
||||
if (wxIsPlatformLittleEndian()) {
|
||||
data=*((wxInt32 *)(&buffer[pos]));
|
||||
pos+=4;
|
||||
} else {
|
||||
data=((wxUint8 *)buffer)[pos++];
|
||||
data|=((wxUint8 *)buffer)[pos++] << 8;
|
||||
data|=((wxUint8 *)buffer)[pos++] << 16;
|
||||
data|=((wxInt8 *)buffer)[pos++] << 24;
|
||||
|
||||
}
|
||||
return true;
|
||||
}
|
||||
bool BinaryFile::Unpack(wxInt64 & data)
|
||||
{
|
||||
if (pos+7>=size) return false;
|
||||
|
||||
if (wxIsPlatformLittleEndian()) {
|
||||
data=*((wxInt64 *)(&buffer[pos]));
|
||||
pos+=8;
|
||||
} else {
|
||||
//for (int i=7;i>=0;i--) data=(data << 8) | ((wxUint8 *)buffer))[pos+i];
|
||||
// pos+=8;
|
||||
data=wxInt64(((wxUint8 *)buffer)[pos++]);
|
||||
data|=wxInt64(((wxUint8 *)buffer)[pos++]) << 8;
|
||||
data|=wxInt64(((wxUint8 *)buffer)[pos++]) << 16;
|
||||
data|=wxInt64(((wxUint8 *)buffer)[pos++]) << 24;
|
||||
data|=wxInt64(((wxUint8 *)buffer)[pos++]) << 32;
|
||||
data|=wxInt64(((wxUint8 *)buffer)[pos++]) << 40;
|
||||
data|=wxInt64(((wxUint8 *)buffer)[pos++]) << 48;
|
||||
data|=wxInt64(((wxInt8 *)buffer)[pos++]) << 56;
|
||||
|
||||
}
|
||||
|
||||
return true;
|
||||
}
|
||||
bool BinaryFile::Unpack(wxUint8 & data)
|
||||
{
|
||||
if (pos>=size) return false;
|
||||
data=((wxInt8 *)buffer)[pos++];
|
||||
return true;
|
||||
}
|
||||
bool BinaryFile::Unpack(wxUint16 & data)
|
||||
{
|
||||
if (pos+1>=size) return false;
|
||||
data=wxUint16(((wxUint8 *)buffer)[pos++]);
|
||||
data|=wxUint16(((wxUint8 *)buffer)[pos++]) << 8;
|
||||
|
||||
return true;
|
||||
}
|
||||
bool BinaryFile::Unpack(wxUint32 & data)
|
||||
{
|
||||
if (pos>=size) return false;
|
||||
if (wxIsPlatformLittleEndian()) {
|
||||
data=*((wxUint32 *)(&buffer[pos]));
|
||||
pos+=4;
|
||||
} else {
|
||||
data=wxUint32(((wxUint8 *)buffer)[pos++]);
|
||||
data|=wxUint32(((wxUint8 *)buffer)[pos++]) << 8;
|
||||
data|=wxUint32(((wxUint8 *)buffer)[pos++]) << 16;
|
||||
data|=wxUint32(((wxUint8 *)buffer)[pos++]) << 24;
|
||||
}
|
||||
|
||||
return true;
|
||||
}
|
||||
bool BinaryFile::Unpack(wxUint64 & data)
|
||||
{
|
||||
if (pos>=size) return false;
|
||||
if (wxIsPlatformLittleEndian()) {
|
||||
data=*((wxInt64 *)(&buffer[pos]));
|
||||
pos+=8;
|
||||
} else {
|
||||
//for (int i=7;i>=0;i--) data=(data << 8) | ((wxUint8 *)buffer))[pos+i];
|
||||
// pos+=8;
|
||||
data=wxUint64(((wxUint8 *)buffer)[pos++]);
|
||||
data|=wxUint64(((wxUint8 *)buffer)[pos++]) << 8;
|
||||
data|=wxUint64(((wxUint8 *)buffer)[pos++]) << 16;
|
||||
data|=wxUint64(((wxUint8 *)buffer)[pos++]) << 24;
|
||||
data|=wxUint64(((wxUint8 *)buffer)[pos++]) << 32;
|
||||
data|=wxUint64(((wxUint8 *)buffer)[pos++]) << 40;
|
||||
data|=wxUint64(((wxUint8 *)buffer)[pos++]) << 48;
|
||||
data|=wxUint64(((wxUint8 *)buffer)[pos++]) << 56;
|
||||
}
|
||||
|
||||
return true;
|
||||
}
|
||||
bool BinaryFile::Unpack(float & data)
|
||||
{
|
||||
if ((pos+4)>=size) return false;
|
||||
if (wxIsPlatformLittleEndian()) {
|
||||
data=*((float *)(&buffer[pos]));
|
||||
pos+=4;
|
||||
} else {
|
||||
unsigned char b[4];
|
||||
for (int i=0; i<4; i++) {
|
||||
b[3-i]=buffer[pos+i];
|
||||
}
|
||||
data=*((float *)(b));
|
||||
pos+=4;
|
||||
}
|
||||
|
||||
return true;
|
||||
}
|
||||
bool BinaryFile::Unpack(double & data)
|
||||
{
|
||||
if ((pos+7)>=size) return false;
|
||||
if (wxIsPlatformLittleEndian()) {
|
||||
data=*((double *)(&buffer[pos]));
|
||||
pos+=8;
|
||||
} else {
|
||||
unsigned char b[8];
|
||||
for (int i=0; i<8; i++) {
|
||||
b[7-i]=buffer[pos+i];
|
||||
}
|
||||
data=*((double *)(b));
|
||||
pos+=8;
|
||||
}
|
||||
|
||||
return true;
|
||||
}
|
||||
bool BinaryFile::Unpack(wxString & data)
|
||||
{
|
||||
wxInt16 i16;
|
||||
if (!Unpack(i16)) return false;
|
||||
if ((pos+i16)>=size) return false;
|
||||
data=wxT("");
|
||||
for (int i=0; i<i16; i++) {
|
||||
data+=wxChar(buffer[pos++]);
|
||||
}
|
||||
return true;
|
||||
}
|
||||
bool BinaryFile::Unpack(wxDateTime & data)
|
||||
{
|
||||
wxInt32 i32;
|
||||
if (!Unpack(i32)) return false;
|
||||
time_t t=i32;
|
||||
data.Set(t);
|
||||
return true;
|
||||
}
|
||||
|
||||
|
||||
bool BinaryFile::Pack(const wxInt8 &data)
|
||||
{
|
||||
f.Write(&data,1);
|
||||
pos++;
|
||||
return true;
|
||||
}
|
||||
bool BinaryFile::Pack(const bool &data)
|
||||
{
|
||||
f.Write((wxInt8*)&data,1);
|
||||
pos++;
|
||||
return true;
|
||||
}
|
||||
bool BinaryFile::Pack(const wxInt16 & data)
|
||||
{
|
||||
wxInt8 *d=(wxInt8 *)&data;
|
||||
f.Write(&d[0],1);
|
||||
f.Write(&d[1],1);
|
||||
pos+=2;
|
||||
return true;
|
||||
}
|
||||
bool BinaryFile::Pack(const wxInt32 & data)
|
||||
{
|
||||
wxInt8 *d=(wxInt8 *)&data;
|
||||
for (int i=0; i<4; i++) f.Write(&d[i],1);
|
||||
pos+=4;
|
||||
return true;
|
||||
}
|
||||
bool BinaryFile::Pack(const wxInt64 & data)
|
||||
{
|
||||
wxInt8 *d=(wxInt8 *)&data;
|
||||
for (int i=0; i<8; i++) f.Write(&d[i],1);
|
||||
pos+=8;
|
||||
return true;
|
||||
}
|
||||
bool BinaryFile::Pack(const wxUint8 & data)
|
||||
{
|
||||
f.Write(&data,1);
|
||||
pos++;
|
||||
return true;
|
||||
}
|
||||
bool BinaryFile::Pack(const wxUint16 & data)
|
||||
{
|
||||
wxUint8 *d=(wxUint8 *)&data;
|
||||
f.Write(&d[0],1);
|
||||
f.Write(&d[1],1);
|
||||
pos+=2;
|
||||
|
||||
return true;
|
||||
}
|
||||
bool BinaryFile::Pack(const wxUint32 & data)
|
||||
{
|
||||
wxUint8 *d=(wxUint8 *)&data;
|
||||
for (int i=0; i<4; i++) f.Write(&d[i],1);
|
||||
pos+=4;
|
||||
return true;
|
||||
}
|
||||
bool BinaryFile::Pack(const wxUint64 & data)
|
||||
{
|
||||
wxUint8 *d=(wxUint8 *)&data;
|
||||
for (int i=0; i<8; i++) f.Write(&d[i],1);
|
||||
pos+=8;
|
||||
return true;
|
||||
}
|
||||
bool BinaryFile::Pack(const float & data)
|
||||
{
|
||||
wxUint8 *d=(wxUint8 *)&data;
|
||||
for (int i=0; i<4; i++) f.Write(&d[i],1);
|
||||
pos+=4;
|
||||
return true;
|
||||
}
|
||||
bool BinaryFile::Pack(const double & data)
|
||||
{
|
||||
wxUint8 *d=(wxUint8 *)&data;
|
||||
for (int i=0; i<8; i++) f.Write(&d[i],1);
|
||||
pos+=8;
|
||||
return true;
|
||||
}
|
||||
bool BinaryFile::Pack(const wxString & data)
|
||||
{
|
||||
char *s=strdup(data.mb_str());
|
||||
|
||||
wxInt16 i16=strlen(s);
|
||||
Pack(i16);
|
||||
|
||||
for (int i=0; i<i16; i++) {
|
||||
Pack(s[i]);
|
||||
}
|
||||
free(s);
|
||||
|
||||
return true;
|
||||
}
|
||||
bool BinaryFile::Pack(const wxDateTime & data)
|
||||
{
|
||||
time_t t=data.GetTicks();
|
||||
wxInt32 i32=t;
|
||||
Pack(i32);
|
||||
return true;
|
||||
}
|
||||
|
||||
|
80
libs/sleeplib/binary_file.h
Normal file
80
libs/sleeplib/binary_file.h
Normal file
@ -0,0 +1,80 @@
|
||||
/*
|
||||
|
||||
SleepLib BinaryFile Header
|
||||
|
||||
Author: Mark Watkins <jedimark64@users.sourceforge.net>
|
||||
License: GPL
|
||||
|
||||
*/
|
||||
|
||||
#ifndef BINARY_FILE_H
|
||||
#define BINARY_FILE_H
|
||||
|
||||
#include <wx/ffile.h>
|
||||
#include <wx/utils.h>
|
||||
|
||||
enum BFOpenMode { BF_READ, BF_WRITE };
|
||||
const long max_buffer_size=1048576*2;
|
||||
|
||||
class UnpackError
|
||||
{
|
||||
|
||||
};
|
||||
|
||||
class PackBuffer
|
||||
{
|
||||
public:
|
||||
PackBuffer();
|
||||
~PackBuffer();
|
||||
bool Open(wxString filename,BFOpenMode mode);
|
||||
void Close();
|
||||
|
||||
};
|
||||
|
||||
class BinaryFile
|
||||
{
|
||||
public:
|
||||
BinaryFile();
|
||||
~BinaryFile();
|
||||
bool Open(wxString filename,BFOpenMode mode);
|
||||
void Close();
|
||||
|
||||
bool Unpack(bool & data);
|
||||
bool Unpack(wxInt8 & data);
|
||||
bool Unpack(wxInt16 & data);
|
||||
bool Unpack(wxInt32 & data);
|
||||
bool Unpack(wxInt64 & data);
|
||||
bool Unpack(wxUint8 & data);
|
||||
bool Unpack(wxUint16 & data);
|
||||
bool Unpack(wxUint32 & data);
|
||||
bool Unpack(wxUint64 & data);
|
||||
bool Unpack(float & data);
|
||||
bool Unpack(double & data);
|
||||
bool Unpack(wxString & data);
|
||||
bool Unpack(wxDateTime & data);
|
||||
|
||||
bool Pack(const bool &data);
|
||||
bool Pack(const wxInt8 &data);
|
||||
bool Pack(const wxInt16 & data);
|
||||
bool Pack(const wxInt32 & data);
|
||||
bool Pack(const wxInt64 & data);
|
||||
bool Pack(const wxUint8 & data);
|
||||
bool Pack(const wxUint16 & data);
|
||||
bool Pack(const wxUint32 & data);
|
||||
bool Pack(const wxUint64 & data);
|
||||
bool Pack(const float & data);
|
||||
bool Pack(const double & data);
|
||||
bool Pack(const wxString & data);
|
||||
bool Pack(const wxDateTime & data);
|
||||
|
||||
protected:
|
||||
BFOpenMode bf_mode;
|
||||
wxFFile f;
|
||||
char * buffer;
|
||||
int buff_read;
|
||||
long pos;
|
||||
wxString bf_filename;
|
||||
long size;
|
||||
};
|
||||
|
||||
#endif //BINARY_FILE_H
|
1182
libs/sleeplib/machine.cpp
Normal file
1182
libs/sleeplib/machine.cpp
Normal file
File diff suppressed because it is too large
Load Diff
357
libs/sleeplib/machine.h
Normal file
357
libs/sleeplib/machine.h
Normal file
@ -0,0 +1,357 @@
|
||||
/*
|
||||
SleepLib Machine Implementation
|
||||
|
||||
Author: Mark Watkins <jedimark64@users.sourceforge.net>
|
||||
License: GPL
|
||||
*/
|
||||
|
||||
#ifndef MACHINE_H
|
||||
#define MACHINE_H
|
||||
|
||||
#include <wx/string.h>
|
||||
#include <wx/variant.h>
|
||||
#include <wx/dir.h>
|
||||
#include <wx/filename.h>
|
||||
#include <wx/ffile.h>
|
||||
#include <wx/utils.h>
|
||||
#include <map>
|
||||
#include <vector>
|
||||
#include <list>
|
||||
|
||||
#include "tinyxml/tinyxml.h"
|
||||
#include "preferences.h"
|
||||
|
||||
using namespace std;
|
||||
|
||||
typedef long MachineID;
|
||||
typedef long SessionID;
|
||||
|
||||
class BoundsError {};
|
||||
|
||||
enum MachineType: short { MT_UNKNOWN=0,MT_CPAP,MT_OXIMETER,MT_SLEEPSTAGE };
|
||||
// I wish C++ could extend enums.
|
||||
// Could be implimented by MachineLoader's register function requesting a block of integer space,
|
||||
// and a map to name strings.
|
||||
|
||||
// Be cautious when extending these.. add to the end of each groups to preserve file formats.
|
||||
enum MachineCode:
|
||||
wxInt16 {
|
||||
// General Event Codes
|
||||
CPAP_Obstructive=0, CPAP_Hypopnea, CPAP_ClearAirway, CPAP_RERA, CPAP_VSnore, CPAP_FlowLimit,
|
||||
CPAP_Leak, CPAP_Pressure, CPAP_EAP, CPAP_IAP, CPAP_CSR, CPAP_FlowRate,
|
||||
|
||||
// General CPAP Summary Information
|
||||
CPAP_PressureMin=0x80, CPAP_PressureMax, CPAP_RampTime, CPAP_RampStartingPressure, CPAP_Mode, CPAP_PressureReliefType,
|
||||
CPAP_PressureReliefSetting, CPAP_HumidifierSetting, CPAP_HumidifierStatus, CPAP_PressureMinAchieved,
|
||||
CPAP_PressureMaxAchieved, CPAP_PressurePercentValue, CPAP_PressurePercentName, CPAP_PressureAverage, CPAP_PressureMedian,
|
||||
CPAP_LeakMedian,CPAP_LeakMinimum,CPAP_LeakMaximum,CPAP_LeakAverage,
|
||||
|
||||
// PRS1 Specific Codes
|
||||
PRS1_PressurePulse=0x1000, PRS1_VSnore2,
|
||||
PRS1_Unknown00, PRS1_Unknown01, PRS1_Unknown08, PRS1_Unknown09, PRS1_Unknown0B, PRS1_Unknown0E, PRS1_Unknown10, PRS1_Unknown12,
|
||||
PRS1_SystemLockStatus, PRS1_SystemResistanceStatus, PRS1_HoseDiameter, PRS1_AutoOff, PRS1_MaskAlert, PRS1_ShowAHI,
|
||||
|
||||
};
|
||||
|
||||
enum FlagType:
|
||||
short { FT_BAR, FT_DOT, FT_SPAN };
|
||||
|
||||
enum CPAPMode:
|
||||
short {
|
||||
MODE_CPAP=0,MODE_APAP,MODE_BIPAP,MODE_ASV
|
||||
};
|
||||
enum PRTypes:
|
||||
short {
|
||||
PR_NONE=0,PR_CFLEX,PR_CFLEXPLUS,PR_AFLEX,PR_EPR,PR_SMARTFLEX
|
||||
};
|
||||
|
||||
extern map<MachineCode,wxString> DefaultMCShortNames;
|
||||
extern map<PRTypes,wxString> PressureReliefNames;
|
||||
extern map<CPAPMode,wxString> CPAPModeNames;
|
||||
|
||||
// These are types supported by wxVariant class. To retain compatability, add to the end of this list only..
|
||||
enum MCDataType:
|
||||
wxInt8 { MC_bool=0, MC_long, MC_float, MC_double, MC_string, MC_datetime };
|
||||
|
||||
typedef wxInt16 SampleFormat;
|
||||
typedef float EventDataType;
|
||||
|
||||
class Session;
|
||||
class Profile;
|
||||
|
||||
class Day
|
||||
{
|
||||
public:
|
||||
Day();
|
||||
~Day();
|
||||
void AddSession(Session *s);
|
||||
|
||||
|
||||
EventDataType min(MachineCode code,int field=0);
|
||||
EventDataType max(MachineCode code,int field=0);
|
||||
EventDataType avg(MachineCode code,int field=0);
|
||||
EventDataType sum(MachineCode code,int field=0);
|
||||
EventDataType count(MachineCode code);
|
||||
EventDataType weighted_avg(MachineCode code,int field=0);
|
||||
|
||||
// Note, the following convert to doubles without considering the consequences fully.
|
||||
EventDataType summary_avg(MachineCode code);
|
||||
EventDataType summary_min(MachineCode code);
|
||||
EventDataType summary_max(MachineCode code);
|
||||
|
||||
const wxDateTime & first(MachineCode code);
|
||||
const wxDateTime & last(MachineCode code);
|
||||
const wxDateTime & first() { return d_first; };
|
||||
const wxDateTime & last() { return d_last; };
|
||||
|
||||
wxTimeSpan total_time();
|
||||
float hours() { return total_time().GetSeconds().GetLo()/3600.0; };
|
||||
|
||||
Session *operator [](int i) { return sessions[i]; };
|
||||
|
||||
vector<Session *>::iterator begin() { return sessions.begin(); };
|
||||
vector<Session *>::iterator end() { return sessions.end(); };
|
||||
|
||||
size_t size() { return sessions.size(); };
|
||||
|
||||
protected:
|
||||
vector<Session *> sessions;
|
||||
wxDateTime d_first,d_last;
|
||||
wxTimeSpan d_totaltime;
|
||||
private:
|
||||
bool d_firstsession;
|
||||
};
|
||||
|
||||
class Machine
|
||||
{
|
||||
public:
|
||||
Machine(Profile *p,MachineID id=0);
|
||||
virtual ~Machine();
|
||||
// virtual bool Open(wxString path){};
|
||||
|
||||
bool Load();
|
||||
bool Save();
|
||||
|
||||
map<wxDateTime,Day *> day;
|
||||
map<SessionID,Session *> sessionlist;
|
||||
map<wxString,wxString> properties;
|
||||
|
||||
Session * SessionExists(SessionID session);
|
||||
void AddSession(Session *s);
|
||||
|
||||
void SetClass(wxString t) {
|
||||
m_class=t;
|
||||
};
|
||||
void SetType(MachineType t) {
|
||||
m_type=t;
|
||||
};
|
||||
const wxString & GetClass() {
|
||||
return m_class;
|
||||
};
|
||||
const MachineType & GetType() {
|
||||
return m_type;
|
||||
};
|
||||
const wxString hexid() {
|
||||
wxString s;
|
||||
s.Printf(wxT("%08lx"),m_id);
|
||||
return s;
|
||||
};
|
||||
const MachineID & id() { return m_id; };
|
||||
const wxDateTime & FirstDay() { return firstday; };
|
||||
const wxDateTime & LastDay() { return lastday; };
|
||||
|
||||
// const wxDateTime & MinTime(const wxDateTime & date);
|
||||
// const wxDateTime & MaxTime(const wxDateTime & date);
|
||||
|
||||
protected:
|
||||
wxDateTime firstday,lastday;
|
||||
|
||||
MachineID m_id;
|
||||
//wxString m_filename;
|
||||
wxString m_class;
|
||||
MachineType m_type;
|
||||
wxString m_path;
|
||||
Profile *profile;
|
||||
bool changed;
|
||||
bool firstsession;
|
||||
};
|
||||
|
||||
|
||||
class Event
|
||||
{
|
||||
friend class Session;
|
||||
public:
|
||||
Event(wxDateTime time,MachineCode code,list<EventDataType> data);
|
||||
~Event();
|
||||
const EventDataType operator[](short i) {
|
||||
if (i<e_fields) return e_data[i];
|
||||
else return 0;
|
||||
};
|
||||
const wxDateTime & time() {
|
||||
return e_time;
|
||||
};
|
||||
const MachineCode & code() {
|
||||
return e_code;
|
||||
};
|
||||
const short & fields() {
|
||||
return e_fields;
|
||||
};
|
||||
protected:
|
||||
wxDateTime e_time;
|
||||
MachineCode e_code;
|
||||
short e_fields;
|
||||
vector<EventDataType> e_data;
|
||||
};
|
||||
|
||||
class Waveform
|
||||
{
|
||||
friend class Session;
|
||||
public:
|
||||
Waveform(wxDateTime time,MachineCode code,SampleFormat * data,int samples,float duration,SampleFormat min, SampleFormat max);
|
||||
~Waveform();
|
||||
const SampleFormat operator[](int i) {
|
||||
if (i<w_samples) return w_data[i];
|
||||
else return 0;
|
||||
};
|
||||
const wxDateTime & start() {
|
||||
return w_time;
|
||||
};
|
||||
const wxDateTime end() {
|
||||
return w_time+w_totalspan;
|
||||
};
|
||||
const MachineCode & code() {
|
||||
return w_code;
|
||||
};
|
||||
const int & samples() {
|
||||
return w_samples;
|
||||
};
|
||||
const float & duration() {
|
||||
return w_duration;
|
||||
};
|
||||
const SampleFormat & min() {
|
||||
return Min;
|
||||
};
|
||||
const SampleFormat & max() {
|
||||
return Max;
|
||||
};
|
||||
|
||||
protected:
|
||||
wxDateTime w_time;
|
||||
MachineCode w_code;
|
||||
SampleFormat * w_data;
|
||||
wxInt32 w_samples;
|
||||
float w_duration;
|
||||
SampleFormat Min,Max;
|
||||
wxTimeSpan w_totalspan;
|
||||
wxTimeSpan w_samplespan;
|
||||
};
|
||||
|
||||
|
||||
class Session
|
||||
{
|
||||
public:
|
||||
Session(Machine *,SessionID);
|
||||
virtual ~Session();
|
||||
|
||||
void AddEvent(Event *e);
|
||||
void AddWaveform(Waveform *w);
|
||||
|
||||
bool Store(wxString path);
|
||||
bool StoreSummary(wxString filename);
|
||||
bool StoreEvents(wxString filename);
|
||||
bool StoreWaveforms(wxString filename);
|
||||
//bool Load(wxString path);
|
||||
bool LoadSummary(wxString filename);
|
||||
bool LoadEvents(wxString filename);
|
||||
bool LoadWaveforms(wxString filename);
|
||||
|
||||
void TrashEvents();
|
||||
void TrashWaveforms();
|
||||
|
||||
const SessionID & session() {
|
||||
return s_session;
|
||||
};
|
||||
const wxDateTime & first() {
|
||||
return s_first;
|
||||
};
|
||||
const wxDateTime & last() {
|
||||
return s_last;
|
||||
};
|
||||
void SetSessionID(SessionID s) {
|
||||
s_session=s;
|
||||
};
|
||||
void set_first(wxDateTime d) {
|
||||
s_first=d;
|
||||
};
|
||||
void set_last(wxDateTime d) {
|
||||
s_last=d;
|
||||
};
|
||||
void set_hours(float h) {
|
||||
s_hours=h;
|
||||
};
|
||||
|
||||
const float & hours() {
|
||||
return s_hours;
|
||||
};
|
||||
int count_events(MachineCode mc) {
|
||||
if (events.find(mc)==events.end()) return 0;
|
||||
return events[mc].size();
|
||||
};
|
||||
double min_event_field(MachineCode mc,int field);
|
||||
double max_event_field(MachineCode mc,int field);
|
||||
double sum_event_field(MachineCode mc,int field);
|
||||
double avg_event_field(MachineCode mc,int field);
|
||||
double weighted_avg_event_field(MachineCode mc,int field);
|
||||
|
||||
map<MachineCode,wxVariant> summary;
|
||||
void SetChanged(bool val) {
|
||||
s_changed=val;
|
||||
};
|
||||
bool IsChanged() {
|
||||
return s_changed;
|
||||
};
|
||||
map<MachineCode,vector<Event *> > events;
|
||||
map<MachineCode,vector<Waveform *> > waveforms;
|
||||
|
||||
protected:
|
||||
SessionID s_session;
|
||||
|
||||
Machine *s_machine;
|
||||
wxDateTime s_first;
|
||||
wxDateTime s_last;
|
||||
float s_hours;
|
||||
bool s_changed;
|
||||
};
|
||||
|
||||
class CPAP:public Machine
|
||||
{
|
||||
public:
|
||||
CPAP(Profile *p,MachineID id=0);
|
||||
virtual ~CPAP();
|
||||
// virtual bool Open(wxString path);
|
||||
map<MachineCode,wxColour> FlagColours;
|
||||
map<MachineCode,FlagType> FlagTypes;
|
||||
list<MachineCode> SleepFlags;
|
||||
};
|
||||
|
||||
class Oximeter:public Machine
|
||||
{
|
||||
public:
|
||||
Oximeter(Profile *p,MachineID id=0);
|
||||
virtual ~Oximeter();
|
||||
// virtual bool Open(wxString path){};
|
||||
protected:
|
||||
};
|
||||
|
||||
class SleepStage:public Machine
|
||||
{
|
||||
public:
|
||||
SleepStage(Profile *p,MachineID id=0);
|
||||
virtual ~SleepStage();
|
||||
// virtual bool Open(wxString path){};
|
||||
protected:
|
||||
};
|
||||
|
||||
|
||||
#endif // MACHINE_H
|
||||
|
42
libs/sleeplib/machine_loader.cpp
Normal file
42
libs/sleeplib/machine_loader.cpp
Normal file
@ -0,0 +1,42 @@
|
||||
/*
|
||||
|
||||
SleepLib Machine Loader Base class Implementation
|
||||
|
||||
Author: Mark Watkins <jedimark64@users.sourceforge.net>
|
||||
License: GPL
|
||||
*/
|
||||
|
||||
#include "machine_loader.h"
|
||||
|
||||
//////////////////////////////////////////////////////////////////////////////////////////
|
||||
// Machine Loader implmementation
|
||||
//////////////////////////////////////////////////////////////////////////////////////////
|
||||
|
||||
list<MachineLoader *> m_loaders;
|
||||
|
||||
void RegisterLoader(MachineLoader *loader)
|
||||
{
|
||||
m_loaders.push_back(loader);
|
||||
}
|
||||
void DestroyLoaders()
|
||||
{
|
||||
for (auto i=m_loaders.begin(); i!=m_loaders.end(); i++) {
|
||||
delete (*i);
|
||||
}
|
||||
m_loaders.clear();
|
||||
}
|
||||
|
||||
MachineLoader::MachineLoader()
|
||||
{
|
||||
|
||||
}
|
||||
MachineLoader::~MachineLoader()
|
||||
{
|
||||
|
||||
}
|
||||
|
||||
list<MachineLoader *> GetLoaders()
|
||||
{
|
||||
return m_loaders;
|
||||
}
|
||||
|
26
libs/sleeplib/machine_loader.h
Normal file
26
libs/sleeplib/machine_loader.h
Normal file
@ -0,0 +1,26 @@
|
||||
/*
|
||||
|
||||
SleepLib MachineLoader Base Class Header
|
||||
|
||||
Author: Mark Watkins <jedimark64@users.sourceforge.net>
|
||||
License: GPL
|
||||
|
||||
*/
|
||||
|
||||
#ifndef MACHINE_LOADER_H
|
||||
#define MACHINE_LOADER_H
|
||||
#include "profiles.h"
|
||||
|
||||
class MachineLoader
|
||||
{
|
||||
public:
|
||||
MachineLoader();
|
||||
virtual ~MachineLoader();
|
||||
virtual bool Open(wxString &,Profile *profile)=0;
|
||||
};
|
||||
|
||||
void RegisterLoader(MachineLoader *loader);
|
||||
void DestroyLoaders();
|
||||
list<MachineLoader *> GetLoaders();
|
||||
|
||||
#endif //MACHINE_LOADER_H
|
233
libs/sleeplib/preferences.cpp
Normal file
233
libs/sleeplib/preferences.cpp
Normal file
@ -0,0 +1,233 @@
|
||||
/*
|
||||
|
||||
SleepLib Preferences Implementation
|
||||
|
||||
Author: Mark Watkins <jedimark64@users.sourceforge.net>
|
||||
License: GPL
|
||||
|
||||
*/
|
||||
|
||||
#include <wx/wx.h>
|
||||
#include <wx/filename.h>
|
||||
#include <wx/string.h>
|
||||
#include <wx/variant.h>
|
||||
#include <wx/stdpaths.h>
|
||||
|
||||
#include "preferences.h"
|
||||
|
||||
|
||||
const wxString & GetAppRoot()
|
||||
{
|
||||
static wxString HomeAppRoot;
|
||||
// wxLogMessage(wxStandardPathsBase::Get().GetUserDataDir());
|
||||
|
||||
//HomeAppRoot=s+wxFileName::GetPathSeparator();
|
||||
|
||||
#if defined(__WXMSW__)
|
||||
// This conveniently maps to unix home directory for now in wine.. Change before release if necessary..
|
||||
HomeAppRoot=wxGetHomeDir()+wxFileName::GetPathSeparator()+wxT("My Documents")+wxFileName::GetPathSeparator()+AppRoot;
|
||||
#elif defined(__UNIX__)
|
||||
HomeAppRoot=wxGetHomeDir()+wxFileName::GetPathSeparator()+AppRoot;
|
||||
#elif defined(__WXMAC__)
|
||||
// I have no idea
|
||||
HomeAppRoot=wxGetHomeDir()+wxFileName::GetPathSeparator()+AppRoot;
|
||||
#endif
|
||||
//HomeAppRoot+=wxFileName::GetPathSeparator(); // Trailing separator
|
||||
return HomeAppRoot;
|
||||
}
|
||||
|
||||
Preferences::Preferences()
|
||||
{
|
||||
p_name=wxT("Preferences");
|
||||
p_path=GetAppRoot();
|
||||
}
|
||||
|
||||
Preferences::Preferences(wxString name,wxString filename)
|
||||
{
|
||||
|
||||
const wxString xmlext=wxT(".xml");
|
||||
const wxString sep=wxFileName::GetPathSeparator();
|
||||
|
||||
if (name.EndsWith(xmlext)) {
|
||||
p_name=name.BeforeLast(wxChar('.'));
|
||||
} else {
|
||||
p_name=name;
|
||||
}
|
||||
|
||||
if (filename.IsEmpty()) {
|
||||
p_filename=GetAppRoot()+sep+p_name+xmlext;
|
||||
} else {
|
||||
if (!filename.Contains(sep)) {
|
||||
p_filename=GetAppRoot()+sep;
|
||||
} else p_filename=wxT("");
|
||||
|
||||
p_filename+=filename;
|
||||
|
||||
if (!p_filename.EndsWith(xmlext)) p_filename+=xmlext;
|
||||
}
|
||||
}
|
||||
|
||||
Preferences::~Preferences()
|
||||
{
|
||||
//Save(); // Don't..Save calls a virtual function.
|
||||
}
|
||||
|
||||
int Preferences::GetCode(wxString s)
|
||||
{
|
||||
int prefcode=0;
|
||||
for (auto i=p_codes.begin(); i!=p_codes.end(); i++) {
|
||||
if (i->second==s) return i->first;
|
||||
prefcode++;
|
||||
}
|
||||
p_codes[prefcode]=s;
|
||||
return prefcode;
|
||||
}
|
||||
|
||||
const wxString Preferences::Get(wxString name)
|
||||
{
|
||||
wxString temp;
|
||||
wxChar obr=wxChar('{');
|
||||
wxChar cbr=wxChar('}');
|
||||
wxString t,a,ref; // How I miss Regular Expressions here..
|
||||
if (p_preferences.find(name)!=p_preferences.end()) {
|
||||
temp=wxT("");
|
||||
t=p_preferences[name].MakeString();
|
||||
if (p_preferences[name].GetType()!=wxT("string")) {
|
||||
return t;
|
||||
}
|
||||
} else {
|
||||
t=name; // parse the string..
|
||||
}
|
||||
while (t.Contains(obr)) {
|
||||
temp+=t.BeforeFirst(obr);
|
||||
a=t.AfterFirst(obr);
|
||||
if (a.StartsWith(wxT("{"))) {
|
||||
temp+=obr;
|
||||
t=a.AfterFirst(obr);
|
||||
continue;
|
||||
}
|
||||
ref=a.BeforeFirst(cbr);
|
||||
|
||||
if (ref.Lower()==wxT("home")) {
|
||||
temp+=GetAppRoot();
|
||||
} else if (ref.Lower()==wxT("user")) {
|
||||
temp+=wxGetUserName();
|
||||
} else if (ref.Lower()==wxT("sep")) {
|
||||
temp+=wxFileName::GetPathSeparator();
|
||||
} else {
|
||||
temp+=Get(ref);
|
||||
}
|
||||
t=a.AfterFirst(cbr);
|
||||
}
|
||||
temp+=t;
|
||||
temp.Replace(wxT("}}"),wxT("}"),true); // Make things look a bit better when escaping braces.
|
||||
|
||||
return temp;
|
||||
}
|
||||
|
||||
bool Preferences::Open(wxString filename)
|
||||
{
|
||||
if (!filename.IsEmpty()) p_filename=filename;
|
||||
|
||||
wxLogVerbose(wxT("Opening ")+p_filename);
|
||||
TiXmlDocument xml(p_filename.mb_str());
|
||||
if (!xml.LoadFile()) {
|
||||
return false;
|
||||
}
|
||||
TiXmlHandle hDoc(&xml);
|
||||
TiXmlElement* pElem;
|
||||
TiXmlHandle hRoot(0);
|
||||
p_preferences.clear();
|
||||
|
||||
pElem=hDoc.FirstChildElement().Element();
|
||||
// should always have a valid root but handle gracefully if it does
|
||||
if (!pElem) return false;
|
||||
|
||||
hRoot=TiXmlHandle(pElem);
|
||||
|
||||
std::map<wxString,wxString> p_types;
|
||||
pElem=hRoot.FirstChild(p_name.mb_str()).FirstChild().Element();
|
||||
for( ; pElem; pElem=pElem->NextSiblingElement()) {
|
||||
|
||||
TiXmlAttribute *attr=pElem->FirstAttribute();
|
||||
assert(attr!=NULL);
|
||||
wxString type(attr->Value(),wxConvUTF8);
|
||||
wxString pKey(pElem->Value(),wxConvUTF8);
|
||||
wxString pText(pElem->GetText(),wxConvUTF8);
|
||||
if (!pKey.IsEmpty() && !pText.IsEmpty()) {
|
||||
if (type==wxT("double")) {
|
||||
double d;
|
||||
pText.ToDouble(&d);
|
||||
p_preferences[pKey]=d;
|
||||
} else if (type==wxT("long")) {
|
||||
long d;
|
||||
pText.ToLong(&d);
|
||||
p_preferences[pKey]=d;
|
||||
} else if (type==wxT("bool")) {
|
||||
long d;
|
||||
pText.ToLong(&d);
|
||||
p_preferences[pKey]=(bool)d;
|
||||
} else if (type==wxT("datetime")) {
|
||||
wxDateTime d;
|
||||
#if wxCHECK_VERSION(2,9,0)
|
||||
wxString::const_iterator end;
|
||||
d.ParseFormat(pText,wxT("%Y-%m-%d %H:%M:%S"),&end);
|
||||
assert(end==pText.end());
|
||||
#else
|
||||
const wxChar *end=d.ParseFormat(pText,wxT("%Y-%m-%d %H:%M:%S"));
|
||||
assert(end!=NULL);
|
||||
#endif
|
||||
p_preferences[pKey]=d;
|
||||
|
||||
} else {
|
||||
p_preferences[pKey]=pText;
|
||||
}
|
||||
}
|
||||
}
|
||||
ExtraLoad(&hRoot);
|
||||
return true;
|
||||
}
|
||||
|
||||
bool Preferences::Save(wxString filename)
|
||||
{
|
||||
if (!filename.IsEmpty()) p_filename=filename;
|
||||
|
||||
TiXmlDocument xml;
|
||||
TiXmlElement* msg;
|
||||
TiXmlComment * comment;
|
||||
TiXmlDeclaration *decl=new TiXmlDeclaration( "1.0", "", "" );
|
||||
xml.LinkEndChild(decl);
|
||||
TiXmlElement *root=new TiXmlElement(AppName.mb_str());
|
||||
xml.LinkEndChild(root);
|
||||
|
||||
if (!p_comment.IsEmpty()) {
|
||||
comment = new TiXmlComment();
|
||||
wxString s=wxT(" ")+p_comment+wxT(" ");
|
||||
comment->SetValue(s.mb_str());
|
||||
root->LinkEndChild(comment);
|
||||
}
|
||||
|
||||
TiXmlElement * msgs = new TiXmlElement(p_name.mb_str());
|
||||
root->LinkEndChild(msgs);
|
||||
for (auto i=p_preferences.begin(); i!=p_preferences.end(); i++) {
|
||||
msg=new TiXmlElement(i->first.mb_str());
|
||||
wxString type=i->second.GetType();
|
||||
msg->SetAttribute("type",type.mb_str());
|
||||
wxString t;
|
||||
|
||||
if (type==wxT("datetime")) {
|
||||
t=i->second.GetDateTime().Format(wxT("%Y-%m-%d %H:%M:%S"));
|
||||
} else {
|
||||
t=i->second.MakeString();
|
||||
}
|
||||
msg->LinkEndChild(new TiXmlText(t.mb_str()));
|
||||
msgs->LinkEndChild(msg);
|
||||
}
|
||||
TiXmlElement *extra=ExtraSave();
|
||||
if (extra) root->LinkEndChild(extra);
|
||||
|
||||
xml.SaveFile(p_filename.mb_str());
|
||||
return true;
|
||||
}
|
||||
|
||||
|
106
libs/sleeplib/preferences.h
Normal file
106
libs/sleeplib/preferences.h
Normal file
@ -0,0 +1,106 @@
|
||||
/*
|
||||
|
||||
SleepLib Preferences Header
|
||||
|
||||
Author: Mark Watkins <jedimark64@users.sourceforge.net>
|
||||
License: GPL
|
||||
|
||||
*/
|
||||
|
||||
#ifndef PREFERENCES_H
|
||||
#define PREFERENCES_H
|
||||
|
||||
#include <wx/string.h>
|
||||
#include <wx/variant.h>
|
||||
#include <map>
|
||||
#include "tinyxml/tinyxml.h"
|
||||
|
||||
|
||||
const wxString AppName=_("Application"); // Outer tag of XML files
|
||||
const wxString AppRoot=wxT("SleepApp"); // The Folder Name
|
||||
|
||||
extern const wxString & GetAppRoot(); //returns app root path plus trailing path separator.
|
||||
|
||||
inline wxString PrefMacro(wxString s)
|
||||
{
|
||||
return wxT("{")+s+wxT("}");
|
||||
};
|
||||
|
||||
|
||||
class Preferences
|
||||
{
|
||||
public:
|
||||
Preferences(wxString name,wxString filename=wxT(""));
|
||||
Preferences();
|
||||
virtual ~Preferences();
|
||||
|
||||
const wxString Get(wxString name);
|
||||
const wxString Get(const char * name) {
|
||||
wxString t(name,wxConvUTF8);
|
||||
return Get(t);
|
||||
};
|
||||
const wxString Get(int code) {
|
||||
return Get(p_codes[code]);
|
||||
};
|
||||
|
||||
// operator[] will not expand {} macros
|
||||
|
||||
wxVariant & operator[](wxString name) {
|
||||
return p_preferences[name];
|
||||
};
|
||||
wxVariant & operator[](const char * name) {
|
||||
wxString t(name,wxConvUTF8);
|
||||
return p_preferences[t];
|
||||
};
|
||||
wxVariant & operator[](int code) {
|
||||
return p_preferences[p_codes[code]];
|
||||
};
|
||||
|
||||
void Set(wxString name,wxVariant value) {
|
||||
p_preferences[name]=value;
|
||||
};
|
||||
void Set(const char * name,wxVariant value) {
|
||||
wxString t(name,wxConvUTF8);
|
||||
p_preferences[t]=value;
|
||||
};
|
||||
void Set(int code,wxVariant value) {
|
||||
Set(p_codes[code],value);
|
||||
};
|
||||
|
||||
bool Exists(wxString name) {
|
||||
return (p_preferences.find(name)!=p_preferences.end());
|
||||
};
|
||||
bool Exists(const char * name) {
|
||||
wxString t(name,wxConvUTF8);
|
||||
return Exists(t);
|
||||
};
|
||||
//bool Exists(int code) { return Exists(p_codes[code]); };
|
||||
|
||||
virtual void ExtraLoad(TiXmlHandle *root) {};
|
||||
virtual TiXmlElement * ExtraSave() {
|
||||
return NULL;
|
||||
};
|
||||
|
||||
virtual bool Open(wxString filename=wxT(""));
|
||||
virtual bool Save(wxString filename=wxT(""));
|
||||
|
||||
void SetComment(const wxString & str) {
|
||||
p_comment=str;
|
||||
};
|
||||
int GetCode(wxString name); // For registering/looking up new preference code.
|
||||
|
||||
std::map<wxString,wxVariant> p_preferences;
|
||||
protected:
|
||||
std::map<int,wxString> p_codes;
|
||||
wxString p_comment;
|
||||
wxString p_name;
|
||||
wxString p_filename;
|
||||
wxString p_path;
|
||||
};
|
||||
|
||||
|
||||
extern Preferences pref;
|
||||
extern Preferences layout;
|
||||
|
||||
#endif // PREFERENCES_H
|
||||
|
252
libs/sleeplib/profiles.cpp
Normal file
252
libs/sleeplib/profiles.cpp
Normal file
@ -0,0 +1,252 @@
|
||||
/*
|
||||
|
||||
SleepLib Profiles Implementation
|
||||
|
||||
Author: Mark Watkins <jedimark64@users.sourceforge.net>
|
||||
License: GPL
|
||||
|
||||
*/
|
||||
#include <wx/filefn.h>
|
||||
#include <wx/filename.h>
|
||||
#include <wx/utils.h>
|
||||
#include <wx/dir.h>
|
||||
#include <wx/log.h>
|
||||
|
||||
#include "preferences.h"
|
||||
#include "profiles.h"
|
||||
#include "machine.h"
|
||||
#include "machine_loader.h"
|
||||
#include "tinyxml/tinyxml.h"
|
||||
|
||||
Preferences *p_pref;
|
||||
Preferences *p_layout;
|
||||
|
||||
Profile::Profile()
|
||||
{
|
||||
p_name=wxT("Profile");
|
||||
p_path=pref.Get("{home}{sep}Profiles");
|
||||
machlist.clear();
|
||||
}
|
||||
Profile::Profile(wxString path)
|
||||
{
|
||||
const wxString xmlext=wxT(".xml");
|
||||
p_name=wxT("Profile");
|
||||
if (path.IsEmpty()) p_path=GetAppRoot();
|
||||
else p_path=path;
|
||||
wxString sep=wxFileName::GetPathSeparator();
|
||||
(*this)["DataFolder"]=p_path;
|
||||
if (!p_path.EndsWith(sep)) p_path+=sep;
|
||||
p_filename=p_path+p_name+xmlext;
|
||||
machlist.clear();
|
||||
}
|
||||
|
||||
Profile::~Profile()
|
||||
{
|
||||
for (auto i=machlist.begin(); i!=machlist.end(); i++) {
|
||||
delete i->second;
|
||||
}
|
||||
}
|
||||
void Profile::LoadMachineData()
|
||||
{
|
||||
for (auto i=machlist.begin(); i!=machlist.end(); i++) {
|
||||
i->second->Load();
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* @brief Machine XML section in profile.
|
||||
* @param root
|
||||
*/
|
||||
void Profile::ExtraLoad(TiXmlHandle *root)
|
||||
{
|
||||
TiXmlElement *elem;
|
||||
elem=root->FirstChild("Machines").FirstChild().Element();
|
||||
if (!elem) {
|
||||
wxLogError(wxT("ExtraLoad: Elem is empty!!!"));
|
||||
}
|
||||
for(; elem; elem=elem->NextSiblingElement()) {
|
||||
wxString pKey(elem->Value(),wxConvUTF8);
|
||||
assert(pKey==wxT("Machine"));
|
||||
|
||||
int m_id;
|
||||
elem->QueryIntAttribute("id",&m_id);
|
||||
int mt;
|
||||
elem->QueryIntAttribute("type",&mt);
|
||||
MachineType m_type=(MachineType)mt;
|
||||
wxString m_class(elem->Attribute("class"),wxConvUTF8);
|
||||
Machine *m;
|
||||
if (m_type==MT_CPAP) m=new CPAP(this,m_id);
|
||||
else if (m_type==MT_OXIMETER) m=new Oximeter(this,m_id);
|
||||
else if (m_type==MT_SLEEPSTAGE) m=new SleepStage(this,m_id);
|
||||
else m=new Machine(this,m_id);
|
||||
m->SetClass(m_class);
|
||||
AddMachine(m);
|
||||
TiXmlElement *e=elem->FirstChildElement();
|
||||
for (; e; e=e->NextSiblingElement()) {
|
||||
//wxString type(attr->Value(),wxConvUTF8);
|
||||
wxString pKey(e->Value(),wxConvUTF8);
|
||||
wxString pText(e->GetText(),wxConvUTF8);
|
||||
m->properties[pKey]=pText;
|
||||
}
|
||||
}
|
||||
}
|
||||
void Profile::AddMachine(Machine *m) {
|
||||
assert(m!=NULL);
|
||||
machlist[m->id()]=m;
|
||||
};
|
||||
|
||||
TiXmlElement * Profile::ExtraSave()
|
||||
{
|
||||
TiXmlElement *mach=new TiXmlElement("Machines");
|
||||
for (auto i=machlist.begin(); i!=machlist.end(); i++) {
|
||||
TiXmlElement *me=new TiXmlElement("Machine");
|
||||
Machine *m=i->second;
|
||||
//wxString t=wxT("0x")+m->hexid();
|
||||
me->SetAttribute("id",m->id());
|
||||
me->SetAttribute("type",m->GetType());
|
||||
me->SetAttribute("class",m->GetClass().mb_str());
|
||||
i->second->properties[wxT("path")]=wxT("{DataFolder}{sep}")+m->hexid();
|
||||
|
||||
for (auto j=i->second->properties.begin(); j!=i->second->properties.end(); j++) {
|
||||
TiXmlElement *mp=new TiXmlElement(j->first.mb_str());
|
||||
mp->LinkEndChild(new TiXmlText(j->second.mb_str()));
|
||||
me->LinkEndChild(mp);
|
||||
}
|
||||
mach->LinkEndChild(me);
|
||||
}
|
||||
//root->LinkEndChild(mach);
|
||||
return mach;
|
||||
|
||||
}
|
||||
|
||||
/**
|
||||
* @brief Import Machine Data
|
||||
* @param path
|
||||
*/
|
||||
void Profile::Import(wxString path)
|
||||
{
|
||||
int c=0;
|
||||
wxLogMessage(wxT("Importing ")+path);
|
||||
list<MachineLoader *>loaders=GetLoaders();
|
||||
for (auto i=loaders.begin(); i!=loaders.end(); i++) {
|
||||
c+=(*i)->Open(path,this);
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
vector<Machine *> Profile::GetMachines(MachineType t)
|
||||
// Returns a vector containing all machine objects regisered of type t
|
||||
{
|
||||
vector<Machine *> vec;
|
||||
map<MachineID,Machine *>::iterator i;
|
||||
|
||||
for (i=machlist.begin(); i!=machlist.end(); i++) {
|
||||
assert(i->second!=NULL);
|
||||
if (i->second->GetType()==t) {
|
||||
vec.push_back(i->second);
|
||||
}
|
||||
}
|
||||
return vec;
|
||||
}
|
||||
|
||||
|
||||
Profile *profile=NULL;
|
||||
wxString SHA1(wxString pass)
|
||||
{
|
||||
return pass;
|
||||
}
|
||||
|
||||
namespace Profiles
|
||||
{
|
||||
|
||||
std::map<wxString,Profile *> profiles;
|
||||
|
||||
void Done()
|
||||
{
|
||||
pref.Save();
|
||||
layout.Save();
|
||||
for (auto i=profiles.begin(); i!=profiles.end(); i++) {
|
||||
i->second->Save();
|
||||
delete i->second;
|
||||
}
|
||||
profiles.clear();
|
||||
delete p_pref;
|
||||
delete p_layout;
|
||||
}
|
||||
|
||||
Profile *Get(wxString name)
|
||||
{
|
||||
return profiles[name];
|
||||
}
|
||||
Profile *Create(wxString name,wxString realname,wxString password)
|
||||
{
|
||||
wxString path=pref.Get("{home}{sep}Profiles");
|
||||
if (!wxDirExists(path)) wxMkdir(path);
|
||||
path+=wxFileName::GetPathSeparator()+name;
|
||||
if (!wxDirExists(path)) wxMkdir(path);
|
||||
Profile *prof=new Profile(path);
|
||||
prof->Open();
|
||||
profiles[name]=prof;
|
||||
prof->Set("Username",name);
|
||||
prof->Set("Realname",realname);
|
||||
if (!password.IsEmpty()) prof->Set("Password",SHA1(password));
|
||||
prof->Set("DataFolder",wxT("{home}{sep}Profiles{sep}{Username}"));
|
||||
return prof;
|
||||
}
|
||||
|
||||
Profile *Get()
|
||||
{
|
||||
return profile;
|
||||
}
|
||||
/**
|
||||
* @brief Scan Profile directory loading user profiles
|
||||
*/
|
||||
|
||||
|
||||
|
||||
void Scan()
|
||||
{
|
||||
p_pref=new Preferences(wxT("Preferences"));
|
||||
p_layout=new Preferences(wxT("Layout"));
|
||||
|
||||
pref.Open();
|
||||
layout.Open();
|
||||
|
||||
wxString path=pref.Get("{home}{sep}Profiles");
|
||||
|
||||
if (!wxDirExists(path)) {
|
||||
wxString tmp=pref.Get("{home}");
|
||||
wxLogMessage(wxT("zzz")+tmp);
|
||||
if (!wxDirExists(tmp)) wxMkdir(tmp);
|
||||
Create(wxGetUserId(),wxGetUserName(),wxT(""));
|
||||
return;
|
||||
}
|
||||
wxDir dir;
|
||||
dir.Open(path);
|
||||
// windows beyatching doesn'topen it.
|
||||
if (!dir.IsOpened()) {
|
||||
wxLogError(wxT("Wierded out opening ")+path);
|
||||
return;
|
||||
}
|
||||
wxString filename;
|
||||
bool cont=dir.GetFirst(&filename);
|
||||
list<wxString> names;
|
||||
while (cont) {
|
||||
names.push_back(filename);
|
||||
cont=dir.GetNext(&filename);
|
||||
}
|
||||
if (names.size()==0) {
|
||||
Create(wxGetUserId(),wxGetUserName(),wxT(""));
|
||||
return;
|
||||
}
|
||||
for (auto i=names.begin(); i!=names.end(); i++) {
|
||||
wxString newpath=path+wxFileName::GetPathSeparator()+filename;
|
||||
Profile *prof=new Profile(newpath);
|
||||
prof->Open();
|
||||
profiles[filename]=prof;
|
||||
}
|
||||
|
||||
}
|
||||
|
||||
}; // namespace Profiles
|
||||
|
66
libs/sleeplib/profiles.h
Normal file
66
libs/sleeplib/profiles.h
Normal file
@ -0,0 +1,66 @@
|
||||
/*
|
||||
|
||||
SleepLib Profiles Header
|
||||
|
||||
Author: Mark Watkins <jedimark64@users.sourceforge.net>
|
||||
License: GPL
|
||||
|
||||
*/
|
||||
|
||||
#ifndef PROFILES_H
|
||||
#define PROFILES_H
|
||||
|
||||
#include <wx/string.h>
|
||||
#include <map>
|
||||
#include "machine.h"
|
||||
#include "preferences.h"
|
||||
#include "tinyxml/tinyxml.h"
|
||||
|
||||
class Machine;
|
||||
/**
|
||||
* @class Profile
|
||||
* @author Mark Watkins
|
||||
* @date 28/04/11
|
||||
* @file profiles.h
|
||||
* @brief User profile system
|
||||
*/
|
||||
class Profile:public Preferences
|
||||
{
|
||||
public:
|
||||
Profile(wxString name);
|
||||
Profile();
|
||||
virtual ~Profile();
|
||||
|
||||
map<MachineID,Machine *> machlist;
|
||||
void AddMachine(Machine *m);
|
||||
void LoadMachineData();
|
||||
void Import(wxString path);
|
||||
vector<Machine *> GetMachines(MachineType t);
|
||||
Machine * GetMachine(MachineType t,wxDateTime date);
|
||||
|
||||
virtual void ExtraLoad(TiXmlHandle *root);
|
||||
virtual TiXmlElement * ExtraSave();
|
||||
};
|
||||
|
||||
extern Preferences *p_pref;
|
||||
extern Preferences *p_layout;
|
||||
#define pref (*p_pref)
|
||||
#define layout (*p_layout)
|
||||
|
||||
extern Profile *profile;
|
||||
|
||||
namespace Profiles
|
||||
{
|
||||
|
||||
extern map<wxString,Profile *> profiles;
|
||||
void Scan(); // Initialize and load Profile
|
||||
void Done(); // Save all Profile objects and clear list
|
||||
|
||||
Profile *Create(wxString name,wxString realname,wxString password);
|
||||
Profile *Get(wxString name);
|
||||
Profile *Get();
|
||||
|
||||
};
|
||||
|
||||
#endif //PROFILES_H
|
||||
|
643
libs/sleeplib/prs1_loader.cpp
Normal file
643
libs/sleeplib/prs1_loader.cpp
Normal file
@ -0,0 +1,643 @@
|
||||
/*
|
||||
|
||||
SleepLib PRS1 Loader Implementation
|
||||
|
||||
Author: Mark Watkins <jedimark64@users.sourceforge.net>
|
||||
License: GPL
|
||||
*/
|
||||
|
||||
#include <wx/dir.h>
|
||||
#include <wx/filename.h>
|
||||
#include <wx/ffile.h>
|
||||
#include <wx/textfile.h>
|
||||
#include <wx/string.h>
|
||||
#include <wx/datetime.h>
|
||||
#include <wx/log.h>
|
||||
#include <wx/progdlg.h>
|
||||
|
||||
#include <initializer_list>
|
||||
|
||||
#include "prs1_loader.h"
|
||||
|
||||
|
||||
extern wxProgressDialog *loader_progress;
|
||||
|
||||
map<int,wxString> ModelMap= {
|
||||
{34,wxT("RemStar Pro with C-Flex+")},
|
||||
{35,wxT("RemStar Auto with A-Flex")}
|
||||
};
|
||||
|
||||
PRS1::PRS1(Profile *p,MachineID id):CPAP(p,id)
|
||||
{
|
||||
m_class=wxT("PRS1");
|
||||
properties[wxT("Brand")]=wxT("Philips Respironics");
|
||||
properties[wxT("Model")]=wxT("System One");
|
||||
|
||||
SleepFlags= { CPAP_RERA, PRS1_VSnore2, CPAP_FlowLimit, CPAP_Hypopnea, CPAP_Obstructive, CPAP_ClearAirway, CPAP_CSR };
|
||||
}
|
||||
PRS1::~PRS1()
|
||||
{
|
||||
|
||||
}
|
||||
|
||||
|
||||
PRS1Loader::PRS1Loader()
|
||||
{
|
||||
}
|
||||
|
||||
PRS1Loader::~PRS1Loader()
|
||||
{
|
||||
for (auto i=PRS1List.begin(); i!=PRS1List.end(); i++) {
|
||||
delete i->second;
|
||||
}
|
||||
}
|
||||
Machine *PRS1Loader::CreateMachine(wxString serial,Profile *profile)
|
||||
{
|
||||
wxLogMessage(wxT("Create Machine")+serial);
|
||||
if (!profile) {
|
||||
wxLogMessage(wxT("No Profile!"));
|
||||
return NULL;
|
||||
|
||||
}
|
||||
vector<Machine *> ml=profile->GetMachines(MT_CPAP);
|
||||
bool found=false;
|
||||
for (auto i=ml.begin(); i!=ml.end(); i++) {
|
||||
if (((*i)->GetClass()==wxT("PRS1")) && ((*i)->properties[wxT("Serial")]==serial)) {
|
||||
PRS1List[serial]=*i; //static_cast<CPAP *>(*i);
|
||||
found=true;
|
||||
break;
|
||||
}
|
||||
}
|
||||
if (found) return PRS1List[serial];
|
||||
|
||||
//assert(PRS1List.find(serial)==PRS1List.end())
|
||||
//wxPuts(wxT("Creating CPAP Machine ")+serial);
|
||||
Machine *m=new PRS1(profile,0);
|
||||
|
||||
PRS1List[serial]=m;
|
||||
profile->AddMachine(m);
|
||||
|
||||
m->properties[wxT("Serial")]=serial;
|
||||
|
||||
return m;
|
||||
}
|
||||
bool PRS1Loader::Open(wxString & path,Profile *profile)
|
||||
{
|
||||
|
||||
wxString newpath;
|
||||
wxString sep=wxFileName::GetPathSeparator();
|
||||
wxString pseries=wxT("P-Series");
|
||||
if (path.Right(pseries.Len()+sep.Len())==pseries) {
|
||||
newpath=path;
|
||||
} else {
|
||||
newpath=path+sep+pseries;
|
||||
}
|
||||
wxDir dir;
|
||||
dir.Open(newpath);
|
||||
if (!dir.IsOpened()) return 0;
|
||||
|
||||
list<wxString> SerialNumbers;
|
||||
list<wxString>::iterator sn;
|
||||
|
||||
wxString filename;
|
||||
bool cont=dir.GetFirst(&filename);
|
||||
|
||||
int c=0;
|
||||
while (cont) {
|
||||
if ((filename[0]=='P') && (isdigit(filename[1])) && (isdigit(filename[2]))) {
|
||||
SerialNumbers.push_back(filename);
|
||||
} else if (filename.Lower()==wxT("last.txt")) { // last.txt points to the current serial number
|
||||
wxTextFile f(newpath+sep+filename);
|
||||
f.Open();
|
||||
last=f.GetFirstLine();
|
||||
last.Strip();
|
||||
f.Close();
|
||||
}
|
||||
|
||||
cont=dir.GetNext(&filename);
|
||||
}
|
||||
|
||||
if (SerialNumbers.empty()) return 0;
|
||||
|
||||
Machine *m;
|
||||
for (sn=SerialNumbers.begin(); sn!=SerialNumbers.end(); sn++) {
|
||||
wxString s=*sn;
|
||||
m=CreateMachine(s,profile);
|
||||
|
||||
if (m) OpenMachine(m,newpath+wxFileName::GetPathSeparator()+(*sn));
|
||||
}
|
||||
|
||||
return PRS1List.size();
|
||||
|
||||
return c;
|
||||
}
|
||||
bool PRS1Loader::ParseProperties(Machine *m,wxString filename)
|
||||
{
|
||||
wxTextFile f(filename);
|
||||
f.Open();
|
||||
if (!f.IsOpened()) return false;
|
||||
|
||||
wxString line;
|
||||
map<wxString,wxString> prop;
|
||||
|
||||
wxString s=f.GetFirstLine();
|
||||
wxChar sep=wxChar('=');
|
||||
wxString key,value;
|
||||
while (!f.Eof()) {
|
||||
key=s.BeforeFirst(sep);
|
||||
if (key==s) continue;
|
||||
value=s.AfterFirst(sep).Strip();
|
||||
if (value==s) continue;
|
||||
prop[key]=value;
|
||||
s=f.GetNextLine();
|
||||
}
|
||||
|
||||
if (prop[wxT("ProductType")].IsNumber()) {
|
||||
long i;
|
||||
prop[wxT("ProductType")].ToLong(&i);
|
||||
if (ModelMap.find(i)!=ModelMap.end()) {
|
||||
m->properties[wxT("SubModel")]=ModelMap[i];
|
||||
}
|
||||
}
|
||||
if (prop[wxT("SerialNumber")]!=m->properties[wxT("Serial")]) {
|
||||
wxLogWarning(wxT("Serial Number in PRS1 properties.txt doesn't match directory structure"));
|
||||
} else prop.erase(wxT("SerialNumber")); // already got it stored.
|
||||
|
||||
for (auto i=prop.begin(); i!=prop.end(); i++) {
|
||||
m->properties[i->first]=i->second;
|
||||
}
|
||||
|
||||
f.Close();
|
||||
return true;
|
||||
}
|
||||
|
||||
int PRS1Loader::OpenMachine(Machine *m,wxString path)
|
||||
{
|
||||
|
||||
wxLogDebug(wxT("Opening PRS1 ")+path);
|
||||
//wxPuts(wxT("opening "+path));
|
||||
wxDir dir;
|
||||
dir.Open(path);
|
||||
if (!dir.IsOpened()) return false;
|
||||
|
||||
wxString pathname,filename;
|
||||
|
||||
bool cont=dir.GetFirst(&filename);
|
||||
list<wxString> paths;
|
||||
|
||||
if(loader_progress) {
|
||||
loader_progress->Update(0);
|
||||
}
|
||||
while (cont) {
|
||||
pathname=path+wxFileName::GetPathSeparator()+filename;
|
||||
if ((filename[0]==wxChar('p')) && (isdigit(filename[1]))) {
|
||||
paths.push_back(pathname);
|
||||
} else if (filename.Lower()==wxT("properties.txt")) {
|
||||
ParseProperties(m,pathname);
|
||||
} else if (filename.Lower()==wxT("e")) {
|
||||
// don't really give a crap about .004 files yet.
|
||||
}
|
||||
if (loader_progress) loader_progress->Pulse();
|
||||
cont=dir.GetNext(&filename);
|
||||
}
|
||||
|
||||
SessionID session;
|
||||
long ext;
|
||||
typedef vector<wxString> StringList;
|
||||
map<SessionID,StringList> sessfiles;
|
||||
int size=paths.size();
|
||||
int cnt=0;
|
||||
for (auto p=paths.begin(); p!=paths.end(); p++) {
|
||||
dir.Open(*p);
|
||||
if (!dir.IsOpened()) continue;;
|
||||
bool cont=dir.GetFirst(&filename);
|
||||
|
||||
while (cont) {
|
||||
wxString ext_s=filename.AfterLast(wxChar('.'));
|
||||
wxString session_s=filename.BeforeLast(wxChar('.'));
|
||||
|
||||
if (!ext_s.IsNumber()) continue;
|
||||
if (!session_s.IsNumber()) continue;
|
||||
|
||||
session_s.ToLong(&session);
|
||||
ext_s.ToLong(&ext);
|
||||
if (sessfiles[session].capacity()==0) sessfiles[session].resize(3);
|
||||
|
||||
wxString fullname=*p+wxFileName::GetPathSeparator()+filename;
|
||||
if (ext==1) {
|
||||
sessfiles[session][0]=fullname;
|
||||
} else if (ext==2) {
|
||||
sessfiles[session][1]=fullname;
|
||||
} else if (ext==5) {
|
||||
sessfiles[session][2]=fullname;
|
||||
}
|
||||
cnt++;
|
||||
if (loader_progress) loader_progress->Pulse(); //Update((float(cnt)/float(size)*25));
|
||||
//if (loader_progress) loader_progress->Update((float(cnt)/float(size)*25.0));
|
||||
cont=dir.GetNext(&filename);
|
||||
}
|
||||
}
|
||||
if (sessfiles.size()==0) return 0;
|
||||
|
||||
size=sessfiles.size();
|
||||
cnt=0;
|
||||
for (auto s=sessfiles.begin(); s!=sessfiles.end(); s++) {
|
||||
session=s->first;
|
||||
cnt++;
|
||||
if (loader_progress) loader_progress->Update(25.0+(float(cnt)/float(size)*25.0));
|
||||
|
||||
if (m->SessionExists(session)) continue;
|
||||
if (!s->second[0]) continue;
|
||||
|
||||
Session *sess=new Session(m,session);
|
||||
if (!OpenSummary(sess,s->second[0])) {
|
||||
wxLogWarning(wxT("PRS1Loader: Could'nt open summary file ")+s->second[0]);
|
||||
|
||||
delete sess;
|
||||
continue;
|
||||
}
|
||||
if (sess->hours()<0.016666667) { // Ignore useless sessions under 1 minute
|
||||
delete sess;
|
||||
continue;
|
||||
}
|
||||
|
||||
//sess->SetSessionID(sess->start().GetTicks());
|
||||
m->AddSession(sess);
|
||||
if (!s->second[1].IsEmpty()) {
|
||||
if (!OpenEvents(sess,s->second[1])) {
|
||||
wxLogWarning(wxT("PRS1Loader: Couldn't open event file ")+s->second[1]);
|
||||
}
|
||||
}
|
||||
|
||||
if (!s->second[2].IsEmpty()) {
|
||||
if (!OpenWaveforms(sess,s->second[2])) {
|
||||
wxLogWarning(wxT("PRS1Loader: Couldn't open event file ")+s->second[2]);
|
||||
}
|
||||
}
|
||||
|
||||
sess->summary[CPAP_CSR]=sess->sum_event_field(CPAP_CSR,0);
|
||||
sess->summary[CPAP_VSnore]=(long)sess->count_events(CPAP_VSnore);
|
||||
sess->summary[PRS1_VSnore2]=sess->sum_event_field(PRS1_VSnore2,0);
|
||||
sess->summary[CPAP_PressureMedian]=sess->avg_event_field(CPAP_Pressure,0);
|
||||
|
||||
sess->summary[CPAP_LeakMinimum]=sess->min_event_field(CPAP_Leak,0);
|
||||
sess->summary[CPAP_LeakMaximum]=sess->max_event_field(CPAP_Leak,0); // should be merged..
|
||||
sess->summary[CPAP_LeakMedian]=sess->avg_event_field(CPAP_Leak,0);
|
||||
sess->summary[CPAP_LeakAverage]=sess->weighted_avg_event_field(CPAP_Leak,0);
|
||||
|
||||
//wxPrintf(sess->start().Format()+wxT(" avgsummary=%.3f avgmine=%.3f\n"),sess->summary[CPAP_PressureAverage].GetDouble(),sess->weighted_avg_event_field(CPAP_Pressure,0));
|
||||
sess->SetChanged(true);
|
||||
}
|
||||
m->Save(); // Save any new sessions to disk in our format
|
||||
if (loader_progress) loader_progress->Update(100);
|
||||
return true;
|
||||
}
|
||||
|
||||
bool PRS1Loader::OpenSummary(Session *session,wxString filename)
|
||||
{
|
||||
int size,sequence,seconds,br;
|
||||
time_t timestamp;
|
||||
unsigned char header[24];
|
||||
unsigned char ext;
|
||||
|
||||
//wxLogMessage(wxT("Opening PRS1 Summary ")+filename);
|
||||
wxFFile f(filename,wxT("rb"));
|
||||
|
||||
if (!f.IsOpened()) return false;
|
||||
|
||||
int hl=16;
|
||||
|
||||
br=f.Read(header,hl);
|
||||
|
||||
if (header[0]!=header[5]) return false;
|
||||
|
||||
sequence=size=timestamp=seconds=ext=0;
|
||||
sequence=(header[10] << 24) | (header[9] << 16) | (header[8] << 8) | header[7];
|
||||
timestamp=(header[14] << 24) | (header[13] << 16) | (header[12] << 8) | header[11];
|
||||
size=(header[2] << 8) | header[1];
|
||||
ext=header[6];
|
||||
|
||||
if (ext!=1) return false;
|
||||
//size|=(header[3]<<16) | (header[4]<<24); // the jury is still out on the 32bitness of one. doesn't matter here anyway.
|
||||
|
||||
size-=(hl+2);
|
||||
|
||||
unsigned char sum=0;
|
||||
for (int i=0; i<hl-1; i++) sum+=header[i];
|
||||
if (sum!=header[hl-1]) return false;
|
||||
|
||||
wxDateTime date(timestamp);
|
||||
//wxDateTime tmpdate=date;
|
||||
//wxLogMessage(date.Format()+wxT(" UTC=")+tmpdate.Format());
|
||||
//int ticks=date.GetTicks();
|
||||
|
||||
|
||||
|
||||
if (!date.IsValid()) return false;
|
||||
|
||||
unsigned char *buffer=(unsigned char *)malloc(size);
|
||||
br=f.Read(buffer,size);
|
||||
if (br<size) {
|
||||
delete buffer;
|
||||
return false;
|
||||
}
|
||||
|
||||
session->set_first(date);
|
||||
double max;
|
||||
session->summary[CPAP_PressureMin]=(double)buffer[0x03]/10.0;
|
||||
session->summary[CPAP_PressureMax]=max=(double)buffer[0x04]/10.0;
|
||||
session->summary[CPAP_RampTime]=(long)buffer[0x06]; // Minutes. Convert to seconds/hours here?
|
||||
session->summary[CPAP_RampStartingPressure]=(double)buffer[0x07]/10.0;
|
||||
|
||||
if (max>0) { // Ignoring bipap until I see some more data.
|
||||
session->summary[CPAP_Mode]=(long)MODE_APAP;
|
||||
} else session->summary[CPAP_Mode]=(long)MODE_CPAP;
|
||||
|
||||
// This is incorrect..
|
||||
if (buffer[0x08] & 0x80) { // Flex Setting
|
||||
if (buffer[0x08] & 0x08) {
|
||||
if (max>0) session->summary[CPAP_PressureReliefType]=(long)PR_AFLEX;
|
||||
else session->summary[CPAP_PressureReliefType]=(long)PR_CFLEXPLUS;
|
||||
} else session->summary[CPAP_PressureReliefType]=(long)PR_CFLEX;
|
||||
} else session->summary[CPAP_PressureReliefType]=(long)PR_NONE;
|
||||
|
||||
session->summary[CPAP_PressureReliefSetting]=(long)buffer[0x08] & 3;
|
||||
session->summary[CPAP_HumidifierSetting]=(long)buffer[0x09]&0x0f;
|
||||
session->summary[CPAP_HumidifierStatus]=(buffer[0x09]&0x80)==0x80;
|
||||
session->summary[PRS1_SystemLockStatus]=(buffer[0x0a]&0x80)==0x80;
|
||||
session->summary[PRS1_SystemResistanceStatus]=(buffer[0x0a]&0x40)==0x40;
|
||||
session->summary[PRS1_HoseDiameter]=(long)((buffer[0x0a]&0x08)?15:22);
|
||||
session->summary[PRS1_AutoOff]=(buffer[0x0c]&0x10)==0x10;
|
||||
session->summary[PRS1_MaskAlert]=(buffer[0x0c]&0x08)==0x08;
|
||||
session->summary[PRS1_ShowAHI]=(buffer[0x0c]&0x04)==0x04;
|
||||
|
||||
int duration=buffer[0x14] | (buffer[0x15] << 8);
|
||||
float hours=float(duration)/3600.0;
|
||||
session->set_hours(hours);
|
||||
|
||||
session->set_last(session->first()+wxTimeSpan::Seconds(duration));
|
||||
|
||||
session->summary[CPAP_PressureMinAchieved]=buffer[0x16]/10.0;
|
||||
session->summary[CPAP_PressureMaxAchieved]=buffer[0x17]/10.0;
|
||||
session->summary[CPAP_PressurePercentValue]=buffer[0x18]/10.0;
|
||||
session->summary[CPAP_PressurePercentName]=90.0;
|
||||
session->summary[CPAP_PressureAverage]=buffer[0x19]/10.0;
|
||||
|
||||
session->summary[CPAP_Obstructive]=(long)buffer[0x1C] | (buffer[0x1D] << 8);
|
||||
session->summary[CPAP_ClearAirway]=(long)buffer[0x20] | (buffer[0x21] << 8);
|
||||
session->summary[CPAP_Hypopnea]=(long)buffer[0x2A] | (buffer[0x2B] << 8);
|
||||
session->summary[CPAP_RERA]=(long)buffer[0x2E] | (buffer[0x2F] << 8);
|
||||
session->summary[CPAP_FlowLimit]=(long)buffer[0x30] | (buffer[0x31] << 8);
|
||||
|
||||
delete buffer;
|
||||
return true;
|
||||
}
|
||||
|
||||
bool PRS1Loader::Parse002(Session *session,unsigned char *buffer,int size,time_t timestamp)
|
||||
{
|
||||
vector<MachineCode> Codes= {
|
||||
PRS1_Unknown00, PRS1_Unknown01, CPAP_Pressure, CPAP_EAP, PRS1_PressurePulse, CPAP_RERA, CPAP_Obstructive, CPAP_ClearAirway,
|
||||
PRS1_Unknown08, PRS1_Unknown09, CPAP_Hypopnea, PRS1_Unknown0B, CPAP_FlowLimit, CPAP_VSnore, PRS1_Unknown0E, CPAP_CSR, PRS1_Unknown10,
|
||||
CPAP_Leak, PRS1_Unknown12
|
||||
};
|
||||
wxDateTime start;
|
||||
start.Set(timestamp);
|
||||
wxDateTime t=start;
|
||||
//t.Set(timestamp);
|
||||
int pos=0;
|
||||
int cnt=0;
|
||||
while (pos<size) {
|
||||
char code=buffer[pos++];
|
||||
short size;
|
||||
if (code!=0x12) {
|
||||
size=buffer[pos] | buffer[pos+1] << 8;
|
||||
pos+=2;
|
||||
t+=wxTimeSpan::Seconds(size);
|
||||
}
|
||||
float data0,data1,data2;
|
||||
MachineCode cpapcode=Codes[(int)code];
|
||||
wxDateTime tt=t;
|
||||
cnt++;
|
||||
switch (code) {
|
||||
case 0x00: // Unknown
|
||||
case 0x01: // Unknown
|
||||
case 0x02: // Pressure
|
||||
data0=buffer[pos++]/10.0;
|
||||
session->AddEvent(new Event(t,cpapcode, {data0}));
|
||||
break;
|
||||
case 0x04: // Pressure Pulse
|
||||
data0=buffer[pos++];
|
||||
session->AddEvent(new Event(t,cpapcode, {data0}));
|
||||
break;
|
||||
|
||||
case 0x05: // RERA
|
||||
case 0x06: // Obstructive Apoanea
|
||||
case 0x07: // Clear Airway
|
||||
case 0x0a: // Hypopnea
|
||||
case 0x0c: // Flow Limitation
|
||||
tt-=wxTimeSpan::Seconds((buffer[pos++])); // Subtract Time Offset
|
||||
session->AddEvent(new Event(tt,cpapcode, {}));
|
||||
break;
|
||||
case 0x0d: // Vibratory Snore
|
||||
session->AddEvent(new Event(t,cpapcode, {}));
|
||||
break;
|
||||
case 0x03: // BIPAP Pressure
|
||||
case 0x0b: // Unknown
|
||||
case 0x11: // Leak Rate
|
||||
data0=buffer[pos++];
|
||||
data1=buffer[pos++];
|
||||
if (code==0x11) {
|
||||
session->AddEvent(new Event(t,cpapcode, {data0}));
|
||||
if (data1>0) {
|
||||
session->AddEvent(new Event(tt,PRS1_VSnore2, {data1}));
|
||||
}
|
||||
} else if (code==0x03) {
|
||||
session->AddEvent(new Event(t,CPAP_EAP, {data0,data1}));
|
||||
session->AddEvent(new Event(t,CPAP_IAP, {data1}));
|
||||
} else {
|
||||
session->AddEvent(new Event(t,cpapcode, {data0,data1}));
|
||||
}
|
||||
break;
|
||||
case 0x0e: // Unknown
|
||||
case 0x10: // Unknown
|
||||
data0=buffer[pos++];
|
||||
data1=buffer[pos++];
|
||||
data2=buffer[pos++];
|
||||
session->AddEvent(new Event(t,cpapcode, {data0,data1,data2}));
|
||||
break;
|
||||
case 0x0f: // Cheyne Stokes Respiration
|
||||
data0=buffer[pos+1]<<8 | buffer[pos];
|
||||
pos+=2;
|
||||
data1=buffer[pos++];
|
||||
tt-=wxTimeSpan::Seconds(data1);
|
||||
session->AddEvent(new Event(tt,cpapcode, {data0,data1}));
|
||||
break;
|
||||
case 0x12: // Summary
|
||||
data0=buffer[pos++];
|
||||
data1=buffer[pos++];
|
||||
data2=buffer[pos+1]<<8 | buffer[pos];
|
||||
pos+=2;
|
||||
session->AddEvent(new Event(t,cpapcode, {data0,data1,data2}));
|
||||
break;
|
||||
default:
|
||||
// ERROR!!!
|
||||
throw exception(); // UnknownCode();
|
||||
break;
|
||||
}
|
||||
}
|
||||
return true;
|
||||
}
|
||||
|
||||
bool PRS1Loader::OpenEvents(Session *session,wxString filename)
|
||||
{
|
||||
int size,sequence,seconds,br;
|
||||
time_t timestamp;
|
||||
unsigned char header[24];
|
||||
unsigned char ext;
|
||||
|
||||
//wxLogMessage(wxT("Opening PRS1 Events ")+filename);
|
||||
wxFFile f(filename,wxT("rb"));
|
||||
|
||||
int hl=16;
|
||||
|
||||
br=f.Read(header,hl);
|
||||
|
||||
if (header[0]!=header[5]) return false;
|
||||
|
||||
sequence=size=timestamp=seconds=ext=0;
|
||||
sequence=(header[10] << 24) | (header[9] << 16) | (header[8] << 8) | header[7];
|
||||
timestamp=(header[14] << 24) | (header[13] << 16) | (header[12] << 8) | header[11];
|
||||
size=(header[2] << 8) | header[1];
|
||||
ext=header[6];
|
||||
|
||||
if (ext!=2) return false;
|
||||
|
||||
//size|=(header[3]<<16) | (header[4]<<24); // the jury is still out on the 32bitness of one. doesn't matter here anyway.
|
||||
|
||||
size-=(hl+2);
|
||||
|
||||
unsigned char sum=0;
|
||||
for (int i=0; i<hl-1; i++) sum+=header[i];
|
||||
if (sum!=header[hl-1]) return false;
|
||||
|
||||
unsigned char *buffer=(unsigned char *)malloc(size);
|
||||
br=f.Read(buffer,size);
|
||||
if (br<size) {
|
||||
delete buffer;
|
||||
return false;
|
||||
}
|
||||
Parse002(session,buffer,size,timestamp);
|
||||
|
||||
delete buffer;
|
||||
return true;
|
||||
}
|
||||
|
||||
bool PRS1Loader::OpenWaveforms(Session *session,wxString filename)
|
||||
{
|
||||
//wxLogMessage(wxT("Opening PRS1 Waveforms ")+filename);
|
||||
|
||||
int size,sequence,seconds,br;
|
||||
unsigned cnt=0;
|
||||
time_t timestamp;
|
||||
|
||||
time_t start=0;
|
||||
unsigned char header[24];
|
||||
unsigned char ext;
|
||||
|
||||
wxFFile f(filename,wxT("rb"));
|
||||
int hl=24;
|
||||
vector<char *> wavedata;
|
||||
vector<int> wavesize;
|
||||
int samples=0;
|
||||
int duration=0;
|
||||
|
||||
while (true) {
|
||||
br=f.Read(header,hl);
|
||||
|
||||
if (br<hl) {
|
||||
if (cnt==0) return false;
|
||||
else break;
|
||||
}
|
||||
|
||||
if (header[0]!=header[5]) return false;
|
||||
|
||||
sequence=size=timestamp=seconds=ext=0;
|
||||
sequence=(header[10] << 24) | (header[9] << 16) | (header[8] << 8) | header[7];
|
||||
timestamp=(header[14] << 24) | (header[13] << 16) | (header[12] << 8) | header[11];
|
||||
size=(header[2] << 8) | header[1];
|
||||
ext=header[6];
|
||||
|
||||
if (!start) start=timestamp;
|
||||
seconds=(header[16] << 8) | header[15];
|
||||
|
||||
size-=(hl+2);
|
||||
|
||||
if (ext!=5) return false;
|
||||
|
||||
unsigned char sum=0;
|
||||
for (int i=0; i<hl-1; i++) sum+=header[i];
|
||||
if (sum!=header[hl-1]) return false;
|
||||
|
||||
char * buffer=new char [size];
|
||||
br=f.Read(buffer,size);
|
||||
if (br<size) {
|
||||
delete [] buffer;
|
||||
break;
|
||||
}
|
||||
wavedata.push_back(buffer);
|
||||
wavesize.push_back(size);
|
||||
cnt++;
|
||||
|
||||
duration+=seconds;
|
||||
samples+=size;
|
||||
|
||||
char chkbuf[2];
|
||||
wxInt16 chksum;
|
||||
br=f.Read(chkbuf,2);
|
||||
if (br<2) return false;
|
||||
chksum=chkbuf[0] << 8 | chkbuf[1];
|
||||
|
||||
}
|
||||
|
||||
if (samples==0) return false;
|
||||
|
||||
//double rate=double(duration)/double(samples);
|
||||
|
||||
SampleFormat *data=new SampleFormat [samples];
|
||||
int s=0;
|
||||
|
||||
SampleFormat min,max,c;
|
||||
bool first=true;
|
||||
//assert (cnt==wavedata.size());
|
||||
cnt=wavedata.size();
|
||||
for (unsigned i=0; i<cnt; i++) {
|
||||
for (int j=0; j<wavesize[i]; j++) {
|
||||
c=wavedata[i][j];
|
||||
if (first) {
|
||||
min=max=c;
|
||||
}
|
||||
if (min>c) min=c;
|
||||
if (max<c) max=c;
|
||||
data[s++]=c;
|
||||
}
|
||||
delete wavedata[i];
|
||||
}
|
||||
Waveform *w=new Waveform(start,CPAP_FlowRate,data,samples,duration,min,max);
|
||||
session->AddWaveform(w);
|
||||
if (sequence==9) {
|
||||
sequence=9;
|
||||
}
|
||||
//wxLogMessage(wxT("Done PRS1 Waveforms ")+filename);
|
||||
return true;
|
||||
}
|
||||
|
||||
|
||||
bool initialized=false;
|
||||
void PRS1Loader::Register()
|
||||
{
|
||||
if (initialized) return;
|
||||
wxLogVerbose(wxT("Registering PRS1Loader"));
|
||||
RegisterLoader(new PRS1Loader());
|
||||
initialized=true;
|
||||
}
|
||||
|
48
libs/sleeplib/prs1_loader.h
Normal file
48
libs/sleeplib/prs1_loader.h
Normal file
@ -0,0 +1,48 @@
|
||||
/*
|
||||
|
||||
SleepLib PRS1 Loader Header
|
||||
|
||||
Author: Mark Watkins <jedimark64@users.sourceforge.net>
|
||||
License: GPL
|
||||
|
||||
*/
|
||||
|
||||
#ifndef PRS1LOADER_H
|
||||
#define PRS1LOADER_H
|
||||
//#include <map>
|
||||
//using namespace std;
|
||||
#include "machine.h" // Base class: MachineLoader
|
||||
#include "machine_loader.h"
|
||||
#include "profiles.h"
|
||||
|
||||
class PRS1:public CPAP
|
||||
{
|
||||
public:
|
||||
PRS1(Profile *p,MachineID id=0);
|
||||
virtual ~PRS1();
|
||||
};
|
||||
|
||||
class PRS1Loader : public MachineLoader
|
||||
{
|
||||
public:
|
||||
PRS1Loader();
|
||||
virtual ~PRS1Loader();
|
||||
virtual bool Open(wxString & path,Profile *profile);
|
||||
|
||||
Machine *CreateMachine(wxString serial,Profile *profile);
|
||||
|
||||
static void Register();
|
||||
protected:
|
||||
wxString last;
|
||||
map<wxString,Machine *> PRS1List;
|
||||
int OpenMachine(Machine *m,wxString path);
|
||||
bool ParseProperties(Machine *m,wxString filename);
|
||||
bool OpenSummary(Session *session,wxString filename);
|
||||
bool OpenEvents(Session *session,wxString filename);
|
||||
bool OpenWaveforms(Session *session,wxString filename);
|
||||
bool Parse002(Session *session,unsigned char *buffer,int size,time_t timestamp);
|
||||
|
||||
};
|
||||
|
||||
|
||||
#endif // PRS1LOADER_H
|
530
libs/tinyxml/readme.txt
Normal file
530
libs/tinyxml/readme.txt
Normal file
@ -0,0 +1,530 @@
|
||||
/** @mainpage
|
||||
|
||||
<h1> TinyXML </h1>
|
||||
|
||||
TinyXML is a simple, small, C++ XML parser that can be easily
|
||||
integrated into other programs.
|
||||
|
||||
<h2> What it does. </h2>
|
||||
|
||||
In brief, TinyXML parses an XML document, and builds from that a
|
||||
Document Object Model (DOM) that can be read, modified, and saved.
|
||||
|
||||
XML stands for "eXtensible Markup Language." It allows you to create
|
||||
your own document markups. Where HTML does a very good job of marking
|
||||
documents for browsers, XML allows you to define any kind of document
|
||||
markup, for example a document that describes a "to do" list for an
|
||||
organizer application. XML is a very structured and convenient format.
|
||||
All those random file formats created to store application data can
|
||||
all be replaced with XML. One parser for everything.
|
||||
|
||||
The best place for the complete, correct, and quite frankly hard to
|
||||
read spec is at <a href="http://www.w3.org/TR/2004/REC-xml-20040204/">
|
||||
http://www.w3.org/TR/2004/REC-xml-20040204/</a>. An intro to XML
|
||||
(that I really like) can be found at
|
||||
<a href="http://skew.org/xml/tutorial/">http://skew.org/xml/tutorial</a>.
|
||||
|
||||
There are different ways to access and interact with XML data.
|
||||
TinyXML uses a Document Object Model (DOM), meaning the XML data is parsed
|
||||
into a C++ objects that can be browsed and manipulated, and then
|
||||
written to disk or another output stream. You can also construct an XML document
|
||||
from scratch with C++ objects and write this to disk or another output
|
||||
stream.
|
||||
|
||||
TinyXML is designed to be easy and fast to learn. It is two headers
|
||||
and four cpp files. Simply add these to your project and off you go.
|
||||
There is an example file - xmltest.cpp - to get you started.
|
||||
|
||||
TinyXML is released under the ZLib license,
|
||||
so you can use it in open source or commercial code. The details
|
||||
of the license are at the top of every source file.
|
||||
|
||||
TinyXML attempts to be a flexible parser, but with truly correct and
|
||||
compliant XML output. TinyXML should compile on any reasonably C++
|
||||
compliant system. It does not rely on exceptions or RTTI. It can be
|
||||
compiled with or without STL support. TinyXML fully supports
|
||||
the UTF-8 encoding, and the first 64k character entities.
|
||||
|
||||
|
||||
<h2> What it doesn't do. </h2>
|
||||
|
||||
TinyXML doesn't parse or use DTDs (Document Type Definitions) or XSLs
|
||||
(eXtensible Stylesheet Language.) There are other parsers out there
|
||||
(check out www.sourceforge.org, search for XML) that are much more fully
|
||||
featured. But they are also much bigger, take longer to set up in
|
||||
your project, have a higher learning curve, and often have a more
|
||||
restrictive license. If you are working with browsers or have more
|
||||
complete XML needs, TinyXML is not the parser for you.
|
||||
|
||||
The following DTD syntax will not parse at this time in TinyXML:
|
||||
|
||||
@verbatim
|
||||
<!DOCTYPE Archiv [
|
||||
<!ELEMENT Comment (#PCDATA)>
|
||||
]>
|
||||
@endverbatim
|
||||
|
||||
because TinyXML sees this as a !DOCTYPE node with an illegally
|
||||
embedded !ELEMENT node. This may be addressed in the future.
|
||||
|
||||
<h2> Tutorials. </h2>
|
||||
|
||||
For the impatient, here is a tutorial to get you going. A great way to get started,
|
||||
but it is worth your time to read this (very short) manual completely.
|
||||
|
||||
- @subpage tutorial0
|
||||
|
||||
<h2> Code Status. </h2>
|
||||
|
||||
TinyXML is mature, tested code. It is very stable. If you find
|
||||
bugs, please file a bug report on the sourceforge web site
|
||||
(www.sourceforge.net/projects/tinyxml). We'll get them straightened
|
||||
out as soon as possible.
|
||||
|
||||
There are some areas of improvement; please check sourceforge if you are
|
||||
interested in working on TinyXML.
|
||||
|
||||
<h2> Related Projects </h2>
|
||||
|
||||
TinyXML projects you may find useful! (Descriptions provided by the projects.)
|
||||
|
||||
<ul>
|
||||
<li> <b>TinyXPath</b> (http://tinyxpath.sourceforge.net). TinyXPath is a small footprint
|
||||
XPath syntax decoder, written in C++.</li>
|
||||
<li> <b>TinyXML++</b> (http://code.google.com/p/ticpp/). TinyXML++ is a completely new
|
||||
interface to TinyXML that uses MANY of the C++ strengths. Templates,
|
||||
exceptions, and much better error handling.</li>
|
||||
</ul>
|
||||
|
||||
<h2> Features </h2>
|
||||
|
||||
<h3> Using STL </h3>
|
||||
|
||||
TinyXML can be compiled to use or not use STL. When using STL, TinyXML
|
||||
uses the std::string class, and fully supports std::istream, std::ostream,
|
||||
operator<<, and operator>>. Many API methods have both 'const char*' and
|
||||
'const std::string&' forms.
|
||||
|
||||
When STL support is compiled out, no STL files are included whatsoever. All
|
||||
the string classes are implemented by TinyXML itself. API methods
|
||||
all use the 'const char*' form for input.
|
||||
|
||||
Use the compile time #define:
|
||||
|
||||
TIXML_USE_STL
|
||||
|
||||
to compile one version or the other. This can be passed by the compiler,
|
||||
or set as the first line of "tinyxml.h".
|
||||
|
||||
Note: If compiling the test code in Linux, setting the environment
|
||||
variable TINYXML_USE_STL=YES/NO will control STL compilation. In the
|
||||
Windows project file, STL and non STL targets are provided. In your project,
|
||||
It's probably easiest to add the line "#define TIXML_USE_STL" as the first
|
||||
line of tinyxml.h.
|
||||
|
||||
<h3> UTF-8 </h3>
|
||||
|
||||
TinyXML supports UTF-8 allowing to manipulate XML files in any language. TinyXML
|
||||
also supports "legacy mode" - the encoding used before UTF-8 support and
|
||||
probably best described as "extended ascii".
|
||||
|
||||
Normally, TinyXML will try to detect the correct encoding and use it. However,
|
||||
by setting the value of TIXML_DEFAULT_ENCODING in the header file, TinyXML
|
||||
can be forced to always use one encoding.
|
||||
|
||||
TinyXML will assume Legacy Mode until one of the following occurs:
|
||||
<ol>
|
||||
<li> If the non-standard but common "UTF-8 lead bytes" (0xef 0xbb 0xbf)
|
||||
begin the file or data stream, TinyXML will read it as UTF-8. </li>
|
||||
<li> If the declaration tag is read, and it has an encoding="UTF-8", then
|
||||
TinyXML will read it as UTF-8. </li>
|
||||
<li> If the declaration tag is read, and it has no encoding specified, then TinyXML will
|
||||
read it as UTF-8. </li>
|
||||
<li> If the declaration tag is read, and it has an encoding="something else", then TinyXML
|
||||
will read it as Legacy Mode. In legacy mode, TinyXML will work as it did before. It's
|
||||
not clear what that mode does exactly, but old content should keep working.</li>
|
||||
<li> Until one of the above criteria is met, TinyXML runs in Legacy Mode.</li>
|
||||
</ol>
|
||||
|
||||
What happens if the encoding is incorrectly set or detected? TinyXML will try
|
||||
to read and pass through text seen as improperly encoded. You may get some strange results or
|
||||
mangled characters. You may want to force TinyXML to the correct mode.
|
||||
|
||||
You may force TinyXML to Legacy Mode by using LoadFile( TIXML_ENCODING_LEGACY ) or
|
||||
LoadFile( filename, TIXML_ENCODING_LEGACY ). You may force it to use legacy mode all
|
||||
the time by setting TIXML_DEFAULT_ENCODING = TIXML_ENCODING_LEGACY. Likewise, you may
|
||||
force it to TIXML_ENCODING_UTF8 with the same technique.
|
||||
|
||||
For English users, using English XML, UTF-8 is the same as low-ASCII. You
|
||||
don't need to be aware of UTF-8 or change your code in any way. You can think
|
||||
of UTF-8 as a "superset" of ASCII.
|
||||
|
||||
UTF-8 is not a double byte format - but it is a standard encoding of Unicode!
|
||||
TinyXML does not use or directly support wchar, TCHAR, or Microsoft's _UNICODE at this time.
|
||||
It is common to see the term "Unicode" improperly refer to UTF-16, a wide byte encoding
|
||||
of unicode. This is a source of confusion.
|
||||
|
||||
For "high-ascii" languages - everything not English, pretty much - TinyXML can
|
||||
handle all languages, at the same time, as long as the XML is encoded
|
||||
in UTF-8. That can be a little tricky, older programs and operating systems
|
||||
tend to use the "default" or "traditional" code page. Many apps (and almost all
|
||||
modern ones) can output UTF-8, but older or stubborn (or just broken) ones
|
||||
still output text in the default code page.
|
||||
|
||||
For example, Japanese systems traditionally use SHIFT-JIS encoding.
|
||||
Text encoded as SHIFT-JIS can not be read by TinyXML.
|
||||
A good text editor can import SHIFT-JIS and then save as UTF-8.
|
||||
|
||||
The <a href="http://skew.org/xml/tutorial/">Skew.org link</a> does a great
|
||||
job covering the encoding issue.
|
||||
|
||||
The test file "utf8test.xml" is an XML containing English, Spanish, Russian,
|
||||
and Simplified Chinese. (Hopefully they are translated correctly). The file
|
||||
"utf8test.gif" is a screen capture of the XML file, rendered in IE. Note that
|
||||
if you don't have the correct fonts (Simplified Chinese or Russian) on your
|
||||
system, you won't see output that matches the GIF file even if you can parse
|
||||
it correctly. Also note that (at least on my Windows machine) console output
|
||||
is in a Western code page, so that Print() or printf() cannot correctly display
|
||||
the file. This is not a bug in TinyXML - just an OS issue. No data is lost or
|
||||
destroyed by TinyXML. The console just doesn't render UTF-8.
|
||||
|
||||
|
||||
<h3> Entities </h3>
|
||||
TinyXML recognizes the pre-defined "character entities", meaning special
|
||||
characters. Namely:
|
||||
|
||||
@verbatim
|
||||
& &
|
||||
< <
|
||||
> >
|
||||
" "
|
||||
' '
|
||||
@endverbatim
|
||||
|
||||
These are recognized when the XML document is read, and translated to there
|
||||
UTF-8 equivalents. For instance, text with the XML of:
|
||||
|
||||
@verbatim
|
||||
Far & Away
|
||||
@endverbatim
|
||||
|
||||
will have the Value() of "Far & Away" when queried from the TiXmlText object,
|
||||
and will be written back to the XML stream/file as an ampersand. Older versions
|
||||
of TinyXML "preserved" character entities, but the newer versions will translate
|
||||
them into characters.
|
||||
|
||||
Additionally, any character can be specified by its Unicode code point:
|
||||
The syntax " " or " " are both to the non-breaking space characher.
|
||||
|
||||
<h3> Printing </h3>
|
||||
TinyXML can print output in several different ways that all have strengths and limitations.
|
||||
|
||||
- Print( FILE* ). Output to a std-C stream, which includes all C files as well as stdout.
|
||||
- "Pretty prints", but you don't have control over printing options.
|
||||
- The output is streamed directly to the FILE object, so there is no memory overhead
|
||||
in the TinyXML code.
|
||||
- used by Print() and SaveFile()
|
||||
|
||||
- operator<<. Output to a c++ stream.
|
||||
- Integrates with standart C++ iostreams.
|
||||
- Outputs in "network printing" mode without line breaks. Good for network transmission
|
||||
and moving XML between C++ objects, but hard for a human to read.
|
||||
|
||||
- TiXmlPrinter. Output to a std::string or memory buffer.
|
||||
- API is less concise
|
||||
- Future printing options will be put here.
|
||||
- Printing may change slightly in future versions as it is refined and expanded.
|
||||
|
||||
<h3> Streams </h3>
|
||||
With TIXML_USE_STL on TinyXML supports C++ streams (operator <<,>>) streams as well
|
||||
as C (FILE*) streams. There are some differences that you may need to be aware of.
|
||||
|
||||
C style output:
|
||||
- based on FILE*
|
||||
- the Print() and SaveFile() methods
|
||||
|
||||
Generates formatted output, with plenty of white space, intended to be as
|
||||
human-readable as possible. They are very fast, and tolerant of ill formed
|
||||
XML documents. For example, an XML document that contains 2 root elements
|
||||
and 2 declarations, will still print.
|
||||
|
||||
C style input:
|
||||
- based on FILE*
|
||||
- the Parse() and LoadFile() methods
|
||||
|
||||
A fast, tolerant read. Use whenever you don't need the C++ streams.
|
||||
|
||||
C++ style output:
|
||||
- based on std::ostream
|
||||
- operator<<
|
||||
|
||||
Generates condensed output, intended for network transmission rather than
|
||||
readability. Depending on your system's implementation of the ostream class,
|
||||
these may be somewhat slower. (Or may not.) Not tolerant of ill formed XML:
|
||||
a document should contain the correct one root element. Additional root level
|
||||
elements will not be streamed out.
|
||||
|
||||
C++ style input:
|
||||
- based on std::istream
|
||||
- operator>>
|
||||
|
||||
Reads XML from a stream, making it useful for network transmission. The tricky
|
||||
part is knowing when the XML document is complete, since there will almost
|
||||
certainly be other data in the stream. TinyXML will assume the XML data is
|
||||
complete after it reads the root element. Put another way, documents that
|
||||
are ill-constructed with more than one root element will not read correctly.
|
||||
Also note that operator>> is somewhat slower than Parse, due to both
|
||||
implementation of the STL and limitations of TinyXML.
|
||||
|
||||
<h3> White space </h3>
|
||||
The world simply does not agree on whether white space should be kept, or condensed.
|
||||
For example, pretend the '_' is a space, and look at "Hello____world". HTML, and
|
||||
at least some XML parsers, will interpret this as "Hello_world". They condense white
|
||||
space. Some XML parsers do not, and will leave it as "Hello____world". (Remember
|
||||
to keep pretending the _ is a space.) Others suggest that __Hello___world__ should become
|
||||
Hello___world.
|
||||
|
||||
It's an issue that hasn't been resolved to my satisfaction. TinyXML supports the
|
||||
first 2 approaches. Call TiXmlBase::SetCondenseWhiteSpace( bool ) to set the desired behavior.
|
||||
The default is to condense white space.
|
||||
|
||||
If you change the default, you should call TiXmlBase::SetCondenseWhiteSpace( bool )
|
||||
before making any calls to Parse XML data, and I don't recommend changing it after
|
||||
it has been set.
|
||||
|
||||
|
||||
<h3> Handles </h3>
|
||||
|
||||
Where browsing an XML document in a robust way, it is important to check
|
||||
for null returns from method calls. An error safe implementation can
|
||||
generate a lot of code like:
|
||||
|
||||
@verbatim
|
||||
TiXmlElement* root = document.FirstChildElement( "Document" );
|
||||
if ( root )
|
||||
{
|
||||
TiXmlElement* element = root->FirstChildElement( "Element" );
|
||||
if ( element )
|
||||
{
|
||||
TiXmlElement* child = element->FirstChildElement( "Child" );
|
||||
if ( child )
|
||||
{
|
||||
TiXmlElement* child2 = child->NextSiblingElement( "Child" );
|
||||
if ( child2 )
|
||||
{
|
||||
// Finally do something useful.
|
||||
@endverbatim
|
||||
|
||||
Handles have been introduced to clean this up. Using the TiXmlHandle class,
|
||||
the previous code reduces to:
|
||||
|
||||
@verbatim
|
||||
TiXmlHandle docHandle( &document );
|
||||
TiXmlElement* child2 = docHandle.FirstChild( "Document" ).FirstChild( "Element" ).Child( "Child", 1 ).ToElement();
|
||||
if ( child2 )
|
||||
{
|
||||
// do something useful
|
||||
@endverbatim
|
||||
|
||||
Which is much easier to deal with. See TiXmlHandle for more information.
|
||||
|
||||
|
||||
<h3> Row and Column tracking </h3>
|
||||
Being able to track nodes and attributes back to their origin location
|
||||
in source files can be very important for some applications. Additionally,
|
||||
knowing where parsing errors occured in the original source can be very
|
||||
time saving.
|
||||
|
||||
TinyXML can tracks the row and column origin of all nodes and attributes
|
||||
in a text file. The TiXmlBase::Row() and TiXmlBase::Column() methods return
|
||||
the origin of the node in the source text. The correct tabs can be
|
||||
configured in TiXmlDocument::SetTabSize().
|
||||
|
||||
|
||||
<h2> Using and Installing </h2>
|
||||
|
||||
To Compile and Run xmltest:
|
||||
|
||||
A Linux Makefile and a Windows Visual C++ .dsw file is provided.
|
||||
Simply compile and run. It will write the file demotest.xml to your
|
||||
disk and generate output on the screen. It also tests walking the
|
||||
DOM by printing out the number of nodes found using different
|
||||
techniques.
|
||||
|
||||
The Linux makefile is very generic and runs on many systems - it
|
||||
is currently tested on mingw and
|
||||
MacOSX. You do not need to run 'make depend'. The dependecies have been
|
||||
hard coded.
|
||||
|
||||
<h3>Windows project file for VC6</h3>
|
||||
<ul>
|
||||
<li>tinyxml: tinyxml library, non-STL </li>
|
||||
<li>tinyxmlSTL: tinyxml library, STL </li>
|
||||
<li>tinyXmlTest: test app, non-STL </li>
|
||||
<li>tinyXmlTestSTL: test app, STL </li>
|
||||
</ul>
|
||||
|
||||
<h3>Makefile</h3>
|
||||
At the top of the makefile you can set:
|
||||
|
||||
PROFILE, DEBUG, and TINYXML_USE_STL. Details (such that they are) are in
|
||||
the makefile.
|
||||
|
||||
In the tinyxml directory, type "make clean" then "make". The executable
|
||||
file 'xmltest' will be created.
|
||||
|
||||
|
||||
|
||||
<h3>To Use in an Application:</h3>
|
||||
|
||||
Add tinyxml.cpp, tinyxml.h, tinyxmlerror.cpp, tinyxmlparser.cpp, tinystr.cpp, and tinystr.h to your
|
||||
project or make file. That's it! It should compile on any reasonably
|
||||
compliant C++ system. You do not need to enable exceptions or
|
||||
RTTI for TinyXML.
|
||||
|
||||
|
||||
<h2> How TinyXML works. </h2>
|
||||
|
||||
An example is probably the best way to go. Take:
|
||||
@verbatim
|
||||
<?xml version="1.0" standalone=no>
|
||||
<!-- Our to do list data -->
|
||||
<ToDo>
|
||||
<Item priority="1"> Go to the <bold>Toy store!</bold></Item>
|
||||
<Item priority="2"> Do bills</Item>
|
||||
</ToDo>
|
||||
@endverbatim
|
||||
|
||||
Its not much of a To Do list, but it will do. To read this file
|
||||
(say "demo.xml") you would create a document, and parse it in:
|
||||
@verbatim
|
||||
TiXmlDocument doc( "demo.xml" );
|
||||
doc.LoadFile();
|
||||
@endverbatim
|
||||
|
||||
And its ready to go. Now lets look at some lines and how they
|
||||
relate to the DOM.
|
||||
|
||||
@verbatim
|
||||
<?xml version="1.0" standalone=no>
|
||||
@endverbatim
|
||||
|
||||
The first line is a declaration, and gets turned into the
|
||||
TiXmlDeclaration class. It will be the first child of the
|
||||
document node.
|
||||
|
||||
This is the only directive/special tag parsed by TinyXML.
|
||||
Generally directive tags are stored in TiXmlUnknown so the
|
||||
commands wont be lost when it is saved back to disk.
|
||||
|
||||
@verbatim
|
||||
<!-- Our to do list data -->
|
||||
@endverbatim
|
||||
|
||||
A comment. Will become a TiXmlComment object.
|
||||
|
||||
@verbatim
|
||||
<ToDo>
|
||||
@endverbatim
|
||||
|
||||
The "ToDo" tag defines a TiXmlElement object. This one does not have
|
||||
any attributes, but does contain 2 other elements.
|
||||
|
||||
@verbatim
|
||||
<Item priority="1">
|
||||
@endverbatim
|
||||
|
||||
Creates another TiXmlElement which is a child of the "ToDo" element.
|
||||
This element has 1 attribute, with the name "priority" and the value
|
||||
"1".
|
||||
|
||||
@verbatim
|
||||
Go to the
|
||||
@endverbatim
|
||||
|
||||
A TiXmlText. This is a leaf node and cannot contain other nodes.
|
||||
It is a child of the "Item" TiXmlElement.
|
||||
|
||||
@verbatim
|
||||
<bold>
|
||||
@endverbatim
|
||||
|
||||
|
||||
Another TiXmlElement, this one a child of the "Item" element.
|
||||
|
||||
Etc.
|
||||
|
||||
Looking at the entire object tree, you end up with:
|
||||
@verbatim
|
||||
TiXmlDocument "demo.xml"
|
||||
TiXmlDeclaration "version='1.0'" "standalone=no"
|
||||
TiXmlComment " Our to do list data"
|
||||
TiXmlElement "ToDo"
|
||||
TiXmlElement "Item" Attribtutes: priority = 1
|
||||
TiXmlText "Go to the "
|
||||
TiXmlElement "bold"
|
||||
TiXmlText "Toy store!"
|
||||
TiXmlElement "Item" Attributes: priority=2
|
||||
TiXmlText "Do bills"
|
||||
@endverbatim
|
||||
|
||||
<h2> Documentation </h2>
|
||||
|
||||
The documentation is build with Doxygen, using the 'dox'
|
||||
configuration file.
|
||||
|
||||
<h2> License </h2>
|
||||
|
||||
TinyXML is released under the zlib license:
|
||||
|
||||
This software is provided 'as-is', without any express or implied
|
||||
warranty. In no event will the authors be held liable for any
|
||||
damages arising from the use of this software.
|
||||
|
||||
Permission is granted to anyone to use this software for any
|
||||
purpose, including commercial applications, and to alter it and
|
||||
redistribute it freely, subject to the following restrictions:
|
||||
|
||||
1. The origin of this software must not be misrepresented; you must
|
||||
not claim that you wrote the original software. If you use this
|
||||
software in a product, an acknowledgment in the product documentation
|
||||
would be appreciated but is not required.
|
||||
|
||||
2. Altered source versions must be plainly marked as such, and
|
||||
must not be misrepresented as being the original software.
|
||||
|
||||
3. This notice may not be removed or altered from any source
|
||||
distribution.
|
||||
|
||||
<h2> References </h2>
|
||||
|
||||
The World Wide Web Consortium is the definitive standard body for
|
||||
XML, and their web pages contain huge amounts of information.
|
||||
|
||||
The definitive spec: <a href="http://www.w3.org/TR/2004/REC-xml-20040204/">
|
||||
http://www.w3.org/TR/2004/REC-xml-20040204/</a>
|
||||
|
||||
I also recommend "XML Pocket Reference" by Robert Eckstein and published by
|
||||
OReilly...the book that got the whole thing started.
|
||||
|
||||
<h2> Contributors, Contacts, and a Brief History </h2>
|
||||
|
||||
Thanks very much to everyone who sends suggestions, bugs, ideas, and
|
||||
encouragement. It all helps, and makes this project fun. A special thanks
|
||||
to the contributors on the web pages that keep it lively.
|
||||
|
||||
So many people have sent in bugs and ideas, that rather than list here
|
||||
we try to give credit due in the "changes.txt" file.
|
||||
|
||||
TinyXML was originally written by Lee Thomason. (Often the "I" still
|
||||
in the documentation.) Lee reviews changes and releases new versions,
|
||||
with the help of Yves Berquin, Andrew Ellerton, and the tinyXml community.
|
||||
|
||||
We appreciate your suggestions, and would love to know if you
|
||||
use TinyXML. Hopefully you will enjoy it and find it useful.
|
||||
Please post questions, comments, file bugs, or contact us at:
|
||||
|
||||
www.sourceforge.net/projects/tinyxml
|
||||
|
||||
Lee Thomason, Yves Berquin, Andrew Ellerton
|
||||
*/
|
112
libs/tinyxml/tinystr.cpp
Normal file
112
libs/tinyxml/tinystr.cpp
Normal file
@ -0,0 +1,112 @@
|
||||
/*
|
||||
www.sourceforge.net/projects/tinyxml
|
||||
Original file by Yves Berquin.
|
||||
|
||||
This software is provided 'as-is', without any express or implied
|
||||
warranty. In no event will the authors be held liable for any
|
||||
damages arising from the use of this software.
|
||||
|
||||
Permission is granted to anyone to use this software for any
|
||||
purpose, including commercial applications, and to alter it and
|
||||
redistribute it freely, subject to the following restrictions:
|
||||
|
||||
1. The origin of this software must not be misrepresented; you must
|
||||
not claim that you wrote the original software. If you use this
|
||||
software in a product, an acknowledgment in the product documentation
|
||||
would be appreciated but is not required.
|
||||
|
||||
2. Altered source versions must be plainly marked as such, and
|
||||
must not be misrepresented as being the original software.
|
||||
|
||||
3. This notice may not be removed or altered from any source
|
||||
distribution.
|
||||
*/
|
||||
|
||||
/*
|
||||
* THIS FILE WAS ALTERED BY Tyge Løvset, 7. April 2005.
|
||||
*/
|
||||
|
||||
|
||||
#ifndef TIXML_USE_STL
|
||||
|
||||
#include "tinystr.h"
|
||||
|
||||
// Error value for find primitive
|
||||
const TiXmlString::size_type TiXmlString::npos = static_cast< TiXmlString::size_type >(-1);
|
||||
|
||||
|
||||
// Null rep.
|
||||
TiXmlString::Rep TiXmlString::nullrep_ = { 0, 0, { '\0' } };
|
||||
|
||||
|
||||
void TiXmlString::reserve (size_type cap)
|
||||
{
|
||||
if (cap > capacity()) {
|
||||
TiXmlString tmp;
|
||||
tmp.init(length(), cap);
|
||||
memcpy(tmp.start(), data(), length());
|
||||
swap(tmp);
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
TiXmlString& TiXmlString::assign(const char* str, size_type len)
|
||||
{
|
||||
size_type cap = capacity();
|
||||
if (len > cap || cap > 3*(len + 8)) {
|
||||
TiXmlString tmp;
|
||||
tmp.init(len);
|
||||
memcpy(tmp.start(), str, len);
|
||||
swap(tmp);
|
||||
} else {
|
||||
memmove(start(), str, len);
|
||||
set_size(len);
|
||||
}
|
||||
return *this;
|
||||
}
|
||||
|
||||
|
||||
TiXmlString& TiXmlString::append(const char* str, size_type len)
|
||||
{
|
||||
size_type newsize = length() + len;
|
||||
if (newsize > capacity()) {
|
||||
reserve (newsize + capacity());
|
||||
}
|
||||
memmove(finish(), str, len);
|
||||
set_size(newsize);
|
||||
return *this;
|
||||
}
|
||||
|
||||
|
||||
TiXmlString operator + (const TiXmlString & a, const TiXmlString & b)
|
||||
{
|
||||
TiXmlString tmp;
|
||||
tmp.reserve(a.length() + b.length());
|
||||
tmp += a;
|
||||
tmp += b;
|
||||
return tmp;
|
||||
}
|
||||
|
||||
TiXmlString operator + (const TiXmlString & a, const char* b)
|
||||
{
|
||||
TiXmlString tmp;
|
||||
TiXmlString::size_type b_len = static_cast<TiXmlString::size_type>( strlen(b) );
|
||||
tmp.reserve(a.length() + b_len);
|
||||
tmp += a;
|
||||
tmp.append(b, b_len);
|
||||
return tmp;
|
||||
}
|
||||
|
||||
TiXmlString operator + (const char* a, const TiXmlString & b)
|
||||
{
|
||||
TiXmlString tmp;
|
||||
TiXmlString::size_type a_len = static_cast<TiXmlString::size_type>( strlen(a) );
|
||||
tmp.reserve(a_len + b.length());
|
||||
tmp.append(a, a_len);
|
||||
tmp += b;
|
||||
return tmp;
|
||||
}
|
||||
|
||||
|
||||
#endif // TIXML_USE_STL
|
||||
|
338
libs/tinyxml/tinystr.h
Normal file
338
libs/tinyxml/tinystr.h
Normal file
@ -0,0 +1,338 @@
|
||||
/*
|
||||
www.sourceforge.net/projects/tinyxml
|
||||
Original file by Yves Berquin.
|
||||
|
||||
This software is provided 'as-is', without any express or implied
|
||||
warranty. In no event will the authors be held liable for any
|
||||
damages arising from the use of this software.
|
||||
|
||||
Permission is granted to anyone to use this software for any
|
||||
purpose, including commercial applications, and to alter it and
|
||||
redistribute it freely, subject to the following restrictions:
|
||||
|
||||
1. The origin of this software must not be misrepresented; you must
|
||||
not claim that you wrote the original software. If you use this
|
||||
software in a product, an acknowledgment in the product documentation
|
||||
would be appreciated but is not required.
|
||||
|
||||
2. Altered source versions must be plainly marked as such, and
|
||||
must not be misrepresented as being the original software.
|
||||
|
||||
3. This notice may not be removed or altered from any source
|
||||
distribution.
|
||||
*/
|
||||
|
||||
/*
|
||||
* THIS FILE WAS ALTERED BY Tyge Lovset, 7. April 2005.
|
||||
*
|
||||
* - completely rewritten. compact, clean, and fast implementation.
|
||||
* - sizeof(TiXmlString) = pointer size (4 bytes on 32-bit systems)
|
||||
* - fixed reserve() to work as per specification.
|
||||
* - fixed buggy compares operator==(), operator<(), and operator>()
|
||||
* - fixed operator+=() to take a const ref argument, following spec.
|
||||
* - added "copy" constructor with length, and most compare operators.
|
||||
* - added swap(), clear(), size(), capacity(), operator+().
|
||||
*/
|
||||
|
||||
#ifndef TIXML_USE_STL
|
||||
|
||||
#ifndef TIXML_STRING_INCLUDED
|
||||
#define TIXML_STRING_INCLUDED
|
||||
|
||||
#include <assert.h>
|
||||
#include <string.h>
|
||||
|
||||
/* The support for explicit isn't that universal, and it isn't really
|
||||
required - it is used to check that the TiXmlString class isn't incorrectly
|
||||
used. Be nice to old compilers and macro it here:
|
||||
*/
|
||||
#if defined(_MSC_VER) && (_MSC_VER >= 1200 )
|
||||
// Microsoft visual studio, version 6 and higher.
|
||||
#define TIXML_EXPLICIT explicit
|
||||
#elif defined(__GNUC__) && (__GNUC__ >= 3 )
|
||||
// GCC version 3 and higher.s
|
||||
#define TIXML_EXPLICIT explicit
|
||||
#else
|
||||
#define TIXML_EXPLICIT
|
||||
#endif
|
||||
|
||||
|
||||
/*
|
||||
TiXmlString is an emulation of a subset of the std::string template.
|
||||
Its purpose is to allow compiling TinyXML on compilers with no or poor STL support.
|
||||
Only the member functions relevant to the TinyXML project have been implemented.
|
||||
The buffer allocation is made by a simplistic power of 2 like mechanism : if we increase
|
||||
a string and there's no more room, we allocate a buffer twice as big as we need.
|
||||
*/
|
||||
class TiXmlString
|
||||
{
|
||||
public :
|
||||
// The size type used
|
||||
typedef size_t size_type;
|
||||
|
||||
// Error value for find primitive
|
||||
static const size_type npos; // = -1;
|
||||
|
||||
|
||||
// TiXmlString empty constructor
|
||||
TiXmlString () : rep_(&nullrep_) {
|
||||
}
|
||||
|
||||
// TiXmlString copy constructor
|
||||
TiXmlString ( const TiXmlString & copy) : rep_(0) {
|
||||
init(copy.length());
|
||||
memcpy(start(), copy.data(), length());
|
||||
}
|
||||
|
||||
// TiXmlString constructor, based on a string
|
||||
TIXML_EXPLICIT TiXmlString ( const char * copy) : rep_(0) {
|
||||
init( static_cast<size_type>( strlen(copy) ));
|
||||
memcpy(start(), copy, length());
|
||||
}
|
||||
|
||||
// TiXmlString constructor, based on a string
|
||||
TIXML_EXPLICIT TiXmlString ( const char * str, size_type len) : rep_(0) {
|
||||
init(len);
|
||||
memcpy(start(), str, len);
|
||||
}
|
||||
|
||||
// TiXmlString destructor
|
||||
~TiXmlString () {
|
||||
quit();
|
||||
}
|
||||
|
||||
// = operator
|
||||
TiXmlString& operator = (const char * copy) {
|
||||
return assign( copy, (size_type)strlen(copy));
|
||||
}
|
||||
|
||||
// = operator
|
||||
TiXmlString& operator = (const TiXmlString & copy) {
|
||||
return assign(copy.start(), copy.length());
|
||||
}
|
||||
|
||||
|
||||
// += operator. Maps to append
|
||||
TiXmlString& operator += (const char * suffix) {
|
||||
return append(suffix, static_cast<size_type>( strlen(suffix) ));
|
||||
}
|
||||
|
||||
// += operator. Maps to append
|
||||
TiXmlString& operator += (char single) {
|
||||
return append(&single, 1);
|
||||
}
|
||||
|
||||
// += operator. Maps to append
|
||||
TiXmlString& operator += (const TiXmlString & suffix) {
|
||||
return append(suffix.data(), suffix.length());
|
||||
}
|
||||
|
||||
|
||||
// Convert a TiXmlString into a null-terminated char *
|
||||
const char * c_str () const {
|
||||
return rep_->str;
|
||||
}
|
||||
|
||||
// Convert a TiXmlString into a char * (need not be null terminated).
|
||||
const char * data () const {
|
||||
return rep_->str;
|
||||
}
|
||||
|
||||
// Return the length of a TiXmlString
|
||||
size_type length () const {
|
||||
return rep_->size;
|
||||
}
|
||||
|
||||
// Alias for length()
|
||||
size_type size () const {
|
||||
return rep_->size;
|
||||
}
|
||||
|
||||
// Checks if a TiXmlString is empty
|
||||
bool empty () const {
|
||||
return rep_->size == 0;
|
||||
}
|
||||
|
||||
// Return capacity of string
|
||||
size_type capacity () const {
|
||||
return rep_->capacity;
|
||||
}
|
||||
|
||||
|
||||
// single char extraction
|
||||
const char& at (size_type index) const {
|
||||
assert( index < length() );
|
||||
return rep_->str[ index ];
|
||||
}
|
||||
|
||||
// [] operator
|
||||
char& operator [] (size_type index) const {
|
||||
assert( index < length() );
|
||||
return rep_->str[ index ];
|
||||
}
|
||||
|
||||
// find a char in a string. Return TiXmlString::npos if not found
|
||||
size_type find (char lookup) const {
|
||||
return find(lookup, 0);
|
||||
}
|
||||
|
||||
// find a char in a string from an offset. Return TiXmlString::npos if not found
|
||||
size_type find (char tofind, size_type offset) const {
|
||||
if (offset >= length()) return npos;
|
||||
|
||||
for (const char* p = c_str() + offset; *p != '\0'; ++p) {
|
||||
if (*p == tofind) return static_cast< size_type >( p - c_str() );
|
||||
}
|
||||
return npos;
|
||||
}
|
||||
|
||||
void clear () {
|
||||
//Lee:
|
||||
//The original was just too strange, though correct:
|
||||
// TiXmlString().swap(*this);
|
||||
//Instead use the quit & re-init:
|
||||
quit();
|
||||
init(0,0);
|
||||
}
|
||||
|
||||
/* Function to reserve a big amount of data when we know we'll need it. Be aware that this
|
||||
function DOES NOT clear the content of the TiXmlString if any exists.
|
||||
*/
|
||||
void reserve (size_type cap);
|
||||
|
||||
TiXmlString& assign (const char* str, size_type len);
|
||||
|
||||
TiXmlString& append (const char* str, size_type len);
|
||||
|
||||
void swap (TiXmlString& other) {
|
||||
Rep* r = rep_;
|
||||
rep_ = other.rep_;
|
||||
other.rep_ = r;
|
||||
}
|
||||
|
||||
private:
|
||||
|
||||
void init(size_type sz) {
|
||||
init(sz, sz);
|
||||
}
|
||||
void set_size(size_type sz) {
|
||||
rep_->str[ rep_->size = sz ] = '\0';
|
||||
}
|
||||
char* start() const {
|
||||
return rep_->str;
|
||||
}
|
||||
char* finish() const {
|
||||
return rep_->str + rep_->size;
|
||||
}
|
||||
|
||||
struct Rep {
|
||||
size_type size, capacity;
|
||||
char str[1];
|
||||
};
|
||||
|
||||
void init(size_type sz, size_type cap) {
|
||||
if (cap) {
|
||||
// Lee: the original form:
|
||||
// rep_ = static_cast<Rep*>(operator new(sizeof(Rep) + cap));
|
||||
// doesn't work in some cases of new being overloaded. Switching
|
||||
// to the normal allocation, although use an 'int' for systems
|
||||
// that are overly picky about structure alignment.
|
||||
const size_type bytesNeeded = sizeof(Rep) + cap;
|
||||
const size_type intsNeeded = ( bytesNeeded + sizeof(int) - 1 ) / sizeof( int );
|
||||
rep_ = reinterpret_cast<Rep*>( new int[ intsNeeded ] );
|
||||
|
||||
rep_->str[ rep_->size = sz ] = '\0';
|
||||
rep_->capacity = cap;
|
||||
} else {
|
||||
rep_ = &nullrep_;
|
||||
}
|
||||
}
|
||||
|
||||
void quit() {
|
||||
if (rep_ != &nullrep_) {
|
||||
// The rep_ is really an array of ints. (see the allocator, above).
|
||||
// Cast it back before delete, so the compiler won't incorrectly call destructors.
|
||||
delete [] ( reinterpret_cast<int*>( rep_ ) );
|
||||
}
|
||||
}
|
||||
|
||||
Rep * rep_;
|
||||
static Rep nullrep_;
|
||||
|
||||
} ;
|
||||
|
||||
|
||||
inline bool operator == (const TiXmlString & a, const TiXmlString & b)
|
||||
{
|
||||
return ( a.length() == b.length() ) // optimization on some platforms
|
||||
&& ( strcmp(a.c_str(), b.c_str()) == 0 ); // actual compare
|
||||
}
|
||||
inline bool operator < (const TiXmlString & a, const TiXmlString & b)
|
||||
{
|
||||
return strcmp(a.c_str(), b.c_str()) < 0;
|
||||
}
|
||||
|
||||
inline bool operator != (const TiXmlString & a, const TiXmlString & b)
|
||||
{
|
||||
return !(a == b);
|
||||
}
|
||||
inline bool operator > (const TiXmlString & a, const TiXmlString & b)
|
||||
{
|
||||
return b < a;
|
||||
}
|
||||
inline bool operator <= (const TiXmlString & a, const TiXmlString & b)
|
||||
{
|
||||
return !(b < a);
|
||||
}
|
||||
inline bool operator >= (const TiXmlString & a, const TiXmlString & b)
|
||||
{
|
||||
return !(a < b);
|
||||
}
|
||||
|
||||
inline bool operator == (const TiXmlString & a, const char* b)
|
||||
{
|
||||
return strcmp(a.c_str(), b) == 0;
|
||||
}
|
||||
inline bool operator == (const char* a, const TiXmlString & b)
|
||||
{
|
||||
return b == a;
|
||||
}
|
||||
inline bool operator != (const TiXmlString & a, const char* b)
|
||||
{
|
||||
return !(a == b);
|
||||
}
|
||||
inline bool operator != (const char* a, const TiXmlString & b)
|
||||
{
|
||||
return !(b == a);
|
||||
}
|
||||
|
||||
TiXmlString operator + (const TiXmlString & a, const TiXmlString & b);
|
||||
TiXmlString operator + (const TiXmlString & a, const char* b);
|
||||
TiXmlString operator + (const char* a, const TiXmlString & b);
|
||||
|
||||
|
||||
/*
|
||||
TiXmlOutStream is an emulation of std::ostream. It is based on TiXmlString.
|
||||
Only the operators that we need for TinyXML have been developped.
|
||||
*/
|
||||
class TiXmlOutStream : public TiXmlString
|
||||
{
|
||||
public :
|
||||
|
||||
// TiXmlOutStream << operator.
|
||||
TiXmlOutStream & operator << (const TiXmlString & in) {
|
||||
*this += in;
|
||||
return *this;
|
||||
}
|
||||
|
||||
// TiXmlOutStream << operator.
|
||||
TiXmlOutStream & operator << (const char * in) {
|
||||
*this += in;
|
||||
return *this;
|
||||
}
|
||||
|
||||
} ;
|
||||
|
||||
#endif // TIXML_STRING_INCLUDED
|
||||
#endif // TIXML_USE_STL
|
||||
|
1748
libs/tinyxml/tinyxml.cpp
Normal file
1748
libs/tinyxml/tinyxml.cpp
Normal file
File diff suppressed because it is too large
Load Diff
2040
libs/tinyxml/tinyxml.h
Normal file
2040
libs/tinyxml/tinyxml.h
Normal file
File diff suppressed because it is too large
Load Diff
52
libs/tinyxml/tinyxmlerror.cpp
Normal file
52
libs/tinyxml/tinyxmlerror.cpp
Normal file
@ -0,0 +1,52 @@
|
||||
/*
|
||||
www.sourceforge.net/projects/tinyxml
|
||||
Original code (2.0 and earlier )copyright (c) 2000-2006 Lee Thomason (www.grinninglizard.com)
|
||||
|
||||
This software is provided 'as-is', without any express or implied
|
||||
warranty. In no event will the authors be held liable for any
|
||||
damages arising from the use of this software.
|
||||
|
||||
Permission is granted to anyone to use this software for any
|
||||
purpose, including commercial applications, and to alter it and
|
||||
redistribute it freely, subject to the following restrictions:
|
||||
|
||||
1. The origin of this software must not be misrepresented; you must
|
||||
not claim that you wrote the original software. If you use this
|
||||
software in a product, an acknowledgment in the product documentation
|
||||
would be appreciated but is not required.
|
||||
|
||||
2. Altered source versions must be plainly marked as such, and
|
||||
must not be misrepresented as being the original software.
|
||||
|
||||
3. This notice may not be removed or altered from any source
|
||||
distribution.
|
||||
*/
|
||||
|
||||
#include "tinyxml.h"
|
||||
|
||||
// The goal of the seperate error file is to make the first
|
||||
// step towards localization. tinyxml (currently) only supports
|
||||
// english error messages, but the could now be translated.
|
||||
//
|
||||
// It also cleans up the code a bit.
|
||||
//
|
||||
|
||||
const char* TiXmlBase::errorString[ TIXML_ERROR_STRING_COUNT ] = {
|
||||
"No error",
|
||||
"Error",
|
||||
"Failed to open file",
|
||||
"Error parsing Element.",
|
||||
"Failed to read Element name",
|
||||
"Error reading Element value.",
|
||||
"Error reading Attributes.",
|
||||
"Error: empty tag.",
|
||||
"Error reading end tag.",
|
||||
"Error parsing Unknown.",
|
||||
"Error parsing Comment.",
|
||||
"Error parsing Declaration.",
|
||||
"Error document empty.",
|
||||
"Error null (0) or unexpected EOF found in input stream.",
|
||||
"Error parsing CDATA.",
|
||||
"Error when TiXmlDocument added to document, because TiXmlDocument can only be at the root.",
|
||||
};
|
||||
|
1457
libs/tinyxml/tinyxmlparser.cpp
Normal file
1457
libs/tinyxml/tinyxmlparser.cpp
Normal file
File diff suppressed because it is too large
Load Diff
33
version.h
Normal file
33
version.h
Normal file
@ -0,0 +1,33 @@
|
||||
#ifndef VERSION_H
|
||||
#define VERSION_H
|
||||
|
||||
namespace AutoVersion{
|
||||
|
||||
//Date Version Types
|
||||
static const char DATE[] = "26";
|
||||
static const char MONTH[] = "05";
|
||||
static const char YEAR[] = "2011";
|
||||
static const char UBUNTU_VERSION_STYLE[] = "11.05";
|
||||
|
||||
//Software Status
|
||||
static const char STATUS[] = "Alpha";
|
||||
static const char STATUS_SHORT[] = "a";
|
||||
|
||||
//Standard Version Type
|
||||
static const long MAJOR = 0;
|
||||
static const long MINOR = 7;
|
||||
static const long BUILD = 1190;
|
||||
static const long REVISION = 940;
|
||||
|
||||
//Miscellaneous Version Types
|
||||
static const long BUILDS_COUNT = 4690;
|
||||
#define RC_FILEVERSION 0,7,1190,940
|
||||
#define RC_FILEVERSION_STRING "0, 7, 1190, 940\0"
|
||||
static const char FULLVERSION_STRING[] = "0.7.1190.940";
|
||||
|
||||
//These values are to keep track of your versioning state, don't modify them.
|
||||
static const long BUILD_HISTORY = 62;
|
||||
|
||||
|
||||
}
|
||||
#endif //VERSION_H
|
Loading…
Reference in New Issue
Block a user