1 /////////////////////////////////////////////////////////////////////////////
3 // Purpose: Document/view classes
4 // Author: Julian Smart
8 // Copyright: (c) Julian Smart and Markus Holzem
9 // Licence: wxWindows licence
10 /////////////////////////////////////////////////////////////////////////////
13 #pragma implementation "docview.h"
16 // For compilers that support precompilation, includes "wx.h".
17 #include "wx/wxprec.h"
27 #if wxUSE_DOC_VIEW_ARCHITECTURE
30 #include "wx/string.h"
34 #include "wx/dialog.h"
37 #include "wx/filedlg.h"
45 #include "wx/msgdlg.h"
46 #include "wx/choicdlg.h"
47 #include "wx/docview.h"
48 #include "wx/printdlg.h"
49 #include "wx/confbase.h"
62 #if !USE_SHARED_LIBRARY
63 IMPLEMENT_ABSTRACT_CLASS(wxDocument
, wxEvtHandler
)
64 IMPLEMENT_ABSTRACT_CLASS(wxView
, wxEvtHandler
)
65 IMPLEMENT_ABSTRACT_CLASS(wxDocTemplate
, wxObject
)
66 IMPLEMENT_DYNAMIC_CLASS(wxDocManager
, wxEvtHandler
)
67 IMPLEMENT_CLASS(wxDocChildFrame
, wxFrame
)
68 IMPLEMENT_CLASS(wxDocParentFrame
, wxFrame
)
69 #if wxUSE_PRINTING_ARCHITECTURE
70 IMPLEMENT_DYNAMIC_CLASS(wxDocPrintout
, wxPrintout
)
72 IMPLEMENT_CLASS(wxCommand
, wxObject
)
73 IMPLEMENT_DYNAMIC_CLASS(wxCommandProcessor
, wxObject
)
74 IMPLEMENT_DYNAMIC_CLASS(wxFileHistory
, wxObject
)
75 // IMPLEMENT_DYNAMIC_CLASS(wxPrintInfo, wxObject)
79 * Definition of wxDocument
82 wxDocument::wxDocument(wxDocument
*parent
)
84 m_documentModified
=FALSE
;
87 m_documentParent
=parent
;
88 m_documentTemplate
= (wxDocTemplate
*) NULL
;
89 m_documentTypeName
= "";
93 bool wxDocument::DeleteContents(void)
98 wxDocument::~wxDocument(void)
102 if (m_commandProcessor
)
103 delete m_commandProcessor
;
105 GetDocumentManager()->RemoveDocument(this);
107 // Not safe to do here, since it'll
108 // invoke virtual view functions expecting to see
109 // valid derived objects: and by the time we get
110 // here, we've called destructors higher up.
114 bool wxDocument::Close(void)
116 if (OnSaveModified())
117 return OnCloseDocument();
122 bool wxDocument::OnCloseDocument(void)
129 // Note that this implicitly deletes the document when
130 // the last view is deleted.
131 bool wxDocument::DeleteAllViews(void)
133 wxNode
*node
= m_documentViews
.First();
136 wxView
*view
= (wxView
*)node
->Data();
140 wxNode
*next
= node
->Next();
142 delete view
; // Deletes node implicitly
148 wxView
*wxDocument::GetFirstView(void) const
150 if (m_documentViews
.Number() == 0)
151 return (wxView
*) NULL
;
152 return (wxView
*)m_documentViews
.First()->Data();
155 wxDocManager
*wxDocument::GetDocumentManager(void) const
157 return m_documentTemplate
->GetDocumentManager();
160 bool wxDocument::OnNewDocument(void)
162 if (!OnSaveModified())
165 if (OnCloseDocument()==FALSE
) return FALSE
;
168 SetDocumentSaved(FALSE
);
171 GetDocumentManager()->MakeDefaultName(name
);
173 SetFilename(name
, TRUE
);
178 bool wxDocument::Save(void)
182 if (!IsModified()) return TRUE
;
183 if (m_documentFile
== "" || !m_savedYet
)
186 ret
= OnSaveDocument(m_documentFile
);
188 SetDocumentSaved(TRUE
);
192 bool wxDocument::SaveAs(void)
194 wxDocTemplate
*docTemplate
= GetDocumentTemplate();
198 char *tmp
= wxFileSelector(_("Save as"), docTemplate
->GetDirectory(), GetFilename(),
199 docTemplate
->GetDefaultExtension(), docTemplate
->GetFileFilter(),
200 wxSAVE
|wxOVERWRITE_PROMPT
, GetDocumentWindow());
206 wxString
fileName(tmp
);
210 wxSplitPath(fileName
, & path
, & name
, & ext
);
212 if (ext
.IsEmpty() || ext
== "")
215 fileName
+= docTemplate
->GetDefaultExtension();
218 SetFilename(fileName
);
219 SetTitle(wxFileNameFromPath(fileName
));
221 GetDocumentManager()->AddFileToHistory(fileName
);
223 // Notify the views that the filename has changed
224 wxNode
*node
= m_documentViews
.First();
227 wxView
*view
= (wxView
*)node
->Data();
228 view
->OnChangeFilename();
232 return OnSaveDocument(m_documentFile
);
235 bool wxDocument::OnSaveDocument(const wxString
& file
)
241 if (wxTheApp
->GetAppName() != "")
242 msgTitle
= wxTheApp
->GetAppName();
244 msgTitle
= wxString(_("File error"));
246 ofstream
store(file
);
247 if (store
.fail() || store
.bad())
249 (void)wxMessageBox(_("Sorry, could not open this file for saving."), msgTitle
, wxOK
| wxICON_EXCLAMATION
,
250 GetDocumentWindow());
254 if (SaveObject(store
)==FALSE
)
256 (void)wxMessageBox(_("Sorry, could not save this file."), msgTitle
, wxOK
| wxICON_EXCLAMATION
,
257 GetDocumentWindow());
266 bool wxDocument::OnOpenDocument(const wxString
& file
)
268 if (!OnSaveModified())
272 if (wxTheApp
->GetAppName() != "")
273 msgTitle
= wxTheApp
->GetAppName();
275 msgTitle
= wxString(_("File error"));
277 ifstream
store(file
);
278 if (store
.fail() || store
.bad())
280 (void)wxMessageBox(_("Sorry, could not open this file."), msgTitle
, wxOK
|wxICON_EXCLAMATION
,
281 GetDocumentWindow());
284 if (LoadObject(store
)==FALSE
)
286 (void)wxMessageBox(_("Sorry, could not open this file."), msgTitle
, wxOK
|wxICON_EXCLAMATION
,
287 GetDocumentWindow());
290 SetFilename(file
, TRUE
);
299 istream
& wxDocument::LoadObject(istream
& stream
)
301 // wxObject::LoadObject(stream);
306 ostream
& wxDocument::SaveObject(ostream
& stream
)
308 // wxObject::SaveObject(stream);
313 bool wxDocument::Revert(void)
319 // Get title, or filename if no title, else unnamed
320 bool wxDocument::GetPrintableName(wxString
& buf
) const
322 if (m_documentTitle
!= "")
324 buf
= m_documentTitle
;
327 else if (m_documentFile
!= "")
329 buf
= wxFileNameFromPath(m_documentFile
);
339 wxWindow
*wxDocument::GetDocumentWindow(void) const
341 wxView
*view
= GetFirstView();
343 return view
->GetFrame();
345 return wxTheApp
->GetTopWindow();
348 wxCommandProcessor
*wxDocument::OnCreateCommandProcessor(void)
350 return new wxCommandProcessor
;
353 // TRUE if safe to close
354 bool wxDocument::OnSaveModified(void)
360 GetPrintableName(title
);
363 if (wxTheApp
->GetAppName() != "")
364 msgTitle
= wxTheApp
->GetAppName();
366 msgTitle
= wxString(_("Warning"));
368 sprintf(buf
, _("Do you want to save changes to document %s?"), (const char *)title
);
369 int res
= wxMessageBox(buf
, msgTitle
, wxYES_NO
|wxCANCEL
|wxICON_QUESTION
,
370 GetDocumentWindow());
376 else if (res
== wxYES
)
378 else if (res
== wxCANCEL
)
384 bool wxDocument::Draw(wxDC
& WXUNUSED(context
))
389 bool wxDocument::AddView(wxView
*view
)
391 if (!m_documentViews
.Member(view
))
393 m_documentViews
.Append(view
);
399 bool wxDocument::RemoveView(wxView
*view
)
401 (void)m_documentViews
.DeleteObject(view
);
406 bool wxDocument::OnCreate(const wxString
& WXUNUSED(path
), long flags
)
408 if (GetDocumentTemplate()->CreateView(this, flags
))
414 // Called after a view is added or removed.
415 // The default implementation deletes the document if
416 // there are no more views.
417 void wxDocument::OnChangedViewList(void)
419 if (m_documentViews
.Number() == 0)
421 if (OnSaveModified())
428 void wxDocument::UpdateAllViews(wxView
*sender
, wxObject
*hint
)
430 wxNode
*node
= m_documentViews
.First();
433 wxView
*view
= (wxView
*)node
->Data();
434 view
->OnUpdate(sender
, hint
);
439 void wxDocument::SetFilename(const wxString
& filename
, bool notifyViews
)
441 m_documentFile
= filename
;
444 // Notify the views that the filename has changed
445 wxNode
*node
= m_documentViews
.First();
448 wxView
*view
= (wxView
*)node
->Data();
449 view
->OnChangeFilename();
460 wxView::wxView(wxDocument
*doc
)
465 m_viewFrame
= (wxFrame
*) NULL
;
468 wxView::~wxView(void)
470 GetDocumentManager()->ActivateView(this, FALSE
, TRUE
);
471 m_viewDocument
->RemoveView(this);
474 // Extend event processing to search the document's event table
475 bool wxView::ProcessEvent(wxEvent
& event
)
477 if ( !GetDocument() || !GetDocument()->ProcessEvent(event
) )
478 return wxEvtHandler::ProcessEvent(event
);
483 void wxView::OnActivateView(bool WXUNUSED(activate
), wxView
*WXUNUSED(activeView
), wxView
*WXUNUSED(deactiveView
))
487 void wxView::OnPrint(wxDC
*dc
, wxObject
*WXUNUSED(info
))
492 void wxView::OnUpdate(wxView
*WXUNUSED(sender
), wxObject
*WXUNUSED(hint
))
496 void wxView::OnChangeFilename(void)
498 if (GetFrame() && GetDocument())
501 GetDocument()->GetPrintableName(name
);
503 GetFrame()->SetTitle(name
);
507 void wxView::SetDocument(wxDocument
*doc
)
509 m_viewDocument
= doc
;
514 bool wxView::Close(bool deleteWindow
)
516 if (OnClose(deleteWindow
))
522 void wxView::Activate(bool activate
)
524 if (GetDocumentManager())
526 OnActivateView(activate
, this, GetDocumentManager()->GetCurrentView());
527 GetDocumentManager()->ActivateView(this, activate
);
531 bool wxView::OnClose(bool WXUNUSED(deleteWindow
))
533 return GetDocument() ? GetDocument()->Close() : TRUE
;
536 #if wxUSE_PRINTING_ARCHITECTURE
537 wxPrintout
*wxView::OnCreatePrintout(void)
539 return new wxDocPrintout(this);
548 wxDocTemplate::wxDocTemplate(wxDocManager
*manager
, const wxString
& descr
,
549 const wxString
& filter
, const wxString
& dir
, const wxString
& ext
,
550 const wxString
& docTypeName
, const wxString
& viewTypeName
,
551 wxClassInfo
*docClassInfo
, wxClassInfo
*viewClassInfo
, long flags
)
553 m_documentManager
= manager
;
555 m_description
= descr
;
558 m_fileFilter
= filter
;
560 m_docTypeName
= docTypeName
;
561 m_viewTypeName
= viewTypeName
;
562 m_documentManager
->AssociateTemplate(this);
564 m_docClassInfo
= docClassInfo
;
565 m_viewClassInfo
= viewClassInfo
;
568 wxDocTemplate::~wxDocTemplate(void)
570 m_documentManager
->DisassociateTemplate(this);
573 // Tries to dynamically construct an object of the right
575 wxDocument
*wxDocTemplate::CreateDocument(const wxString
& path
, long flags
)
578 return (wxDocument
*) NULL
;
579 wxDocument
*doc
= (wxDocument
*)m_docClassInfo
->CreateObject();
580 doc
->SetFilename(path
);
581 doc
->SetDocumentTemplate(this);
582 GetDocumentManager()->AddDocument(doc
);
583 doc
->SetCommandProcessor(doc
->OnCreateCommandProcessor());
585 if (doc
->OnCreate(path
, flags
))
590 return (wxDocument
*) NULL
;
594 wxView
*wxDocTemplate::CreateView(wxDocument
*doc
, long flags
)
596 if (!m_viewClassInfo
)
597 return (wxView
*) NULL
;
598 wxView
*view
= (wxView
*)m_viewClassInfo
->CreateObject();
599 view
->SetDocument(doc
);
600 if (view
->OnCreate(doc
, flags
))
607 return (wxView
*) NULL
;
611 BEGIN_EVENT_TABLE(wxDocManager
, wxEvtHandler
)
612 EVT_MENU(wxID_OPEN
, wxDocManager::OnFileOpen
)
613 EVT_MENU(wxID_CLOSE
, wxDocManager::OnFileClose
)
614 EVT_MENU(wxID_REVERT
, wxDocManager::OnFileRevert
)
615 EVT_MENU(wxID_NEW
, wxDocManager::OnFileNew
)
616 EVT_MENU(wxID_SAVE
, wxDocManager::OnFileSave
)
617 EVT_MENU(wxID_SAVEAS
, wxDocManager::OnFileSaveAs
)
618 EVT_MENU(wxID_UNDO
, wxDocManager::OnUndo
)
619 EVT_MENU(wxID_REDO
, wxDocManager::OnRedo
)
620 EVT_MENU(wxID_PRINT
, wxDocManager::OnPrint
)
621 EVT_MENU(wxID_PRINT_SETUP
, wxDocManager::OnPrintSetup
)
622 EVT_MENU(wxID_PREVIEW
, wxDocManager::OnPreview
)
625 wxDocManager::wxDocManager(long flags
, bool initialize
)
627 m_defaultDocumentNameCounter
= 1;
629 m_currentView
= (wxView
*) NULL
;
630 m_maxDocsOpen
= 10000;
631 m_fileHistory
= (wxFileHistory
*) NULL
;
636 wxDocManager::~wxDocManager(void)
640 delete m_fileHistory
;
643 bool wxDocManager::Clear(bool force
)
645 wxNode
*node
= m_docs
.First();
648 wxDocument
*doc
= (wxDocument
*)node
->Data();
649 wxNode
*next
= node
->Next();
651 if (!doc
->Close() && !force
)
654 // Implicitly deletes the document when the last
655 // view is removed (deleted)
656 doc
->DeleteAllViews();
658 // Check document is deleted
659 if (m_docs
.Member(doc
))
662 // This assumes that documents are not connected in
663 // any way, i.e. deleting one document does NOT
667 node
= m_templates
.First();
670 wxDocTemplate
*templ
= (wxDocTemplate
*) node
->Data();
671 wxNode
* next
= node
->Next();
678 bool wxDocManager::Initialize(void)
680 m_fileHistory
= OnCreateFileHistory();
684 wxFileHistory
*wxDocManager::OnCreateFileHistory(void)
686 return new wxFileHistory
;
689 void wxDocManager::OnFileClose(wxCommandEvent
& WXUNUSED(event
))
691 wxDocument
*doc
= GetCurrentDocument();
696 doc
->DeleteAllViews();
697 if (m_docs
.Member(doc
))
702 void wxDocManager::OnFileNew(wxCommandEvent
& WXUNUSED(event
))
704 CreateDocument(wxString(""), wxDOC_NEW
);
707 void wxDocManager::OnFileOpen(wxCommandEvent
& WXUNUSED(event
))
709 CreateDocument(wxString(""), 0);
712 void wxDocManager::OnFileRevert(wxCommandEvent
& WXUNUSED(event
))
714 wxDocument
*doc
= GetCurrentDocument();
720 void wxDocManager::OnFileSave(wxCommandEvent
& WXUNUSED(event
))
722 wxDocument
*doc
= GetCurrentDocument();
728 void wxDocManager::OnFileSaveAs(wxCommandEvent
& WXUNUSED(event
))
730 wxDocument
*doc
= GetCurrentDocument();
736 void wxDocManager::OnPrint(wxCommandEvent
& WXUNUSED(event
))
738 wxView
*view
= GetCurrentView();
742 wxPrintout
*printout
= view
->OnCreatePrintout();
746 printer
.Print(view
->GetFrame(), printout
, TRUE
);
752 void wxDocManager::OnPrintSetup(wxCommandEvent
& WXUNUSED(event
))
754 wxWindow
*parentWin
= wxTheApp
->GetTopWindow();
755 wxView
*view
= GetCurrentView();
757 parentWin
= view
->GetFrame();
761 wxPrintDialog
printerDialog(parentWin
, & data
);
762 printerDialog
.GetPrintData().SetSetupDialog(TRUE
);
763 printerDialog
.ShowModal();
766 void wxDocManager::OnPreview(wxCommandEvent
& WXUNUSED(event
))
768 wxView
*view
= GetCurrentView();
772 wxPrintout
*printout
= view
->OnCreatePrintout();
775 // Pass two printout objects: for preview, and possible printing.
776 wxPrintPreviewBase
*preview
= (wxPrintPreviewBase
*) NULL
;
777 preview
= new wxPrintPreview(printout
, view
->OnCreatePrintout());
779 wxPreviewFrame
*frame
= new wxPreviewFrame(preview
, (wxFrame
*)wxTheApp
->GetTopWindow(), _("Print Preview"),
780 wxPoint(100, 100), wxSize(600, 650));
781 frame
->Centre(wxBOTH
);
787 void wxDocManager::OnUndo(wxCommandEvent
& WXUNUSED(event
))
789 wxDocument
*doc
= GetCurrentDocument();
792 if (doc
->GetCommandProcessor())
793 doc
->GetCommandProcessor()->Undo();
796 void wxDocManager::OnRedo(wxCommandEvent
& WXUNUSED(event
))
798 wxDocument
*doc
= GetCurrentDocument();
801 if (doc
->GetCommandProcessor())
802 doc
->GetCommandProcessor()->Redo();
805 wxView
*wxDocManager::GetCurrentView(void) const
808 return m_currentView
;
809 if (m_docs
.Number() == 1)
811 wxDocument
* doc
= (wxDocument
*) m_docs
.First()->Data();
812 return doc
->GetFirstView();
814 return (wxView
*) NULL
;
817 // Extend event processing to search the view's event table
818 bool wxDocManager::ProcessEvent(wxEvent
& event
)
820 wxView
* view
= GetCurrentView();
823 if (view
->ProcessEvent(event
))
826 return wxEvtHandler::ProcessEvent(event
);
829 wxDocument
*wxDocManager::CreateDocument(const wxString
& path
, long flags
)
831 wxDocTemplate
**templates
= new wxDocTemplate
*[m_templates
.Number()];
834 for (i
= 0; i
< m_templates
.Number(); i
++)
836 wxDocTemplate
*temp
= (wxDocTemplate
*)(m_templates
.Nth(i
)->Data());
837 if (temp
->IsVisible())
846 return (wxDocument
*) NULL
;
849 // If we've reached the max number of docs, close the
851 if (GetDocuments().Number() >= m_maxDocsOpen
)
853 wxDocument
*doc
= (wxDocument
*)GetDocuments().First()->Data();
856 // Implicitly deletes the document when
857 // the last view is deleted
858 doc
->DeleteAllViews();
860 // Check we're really deleted
861 if (m_docs
.Member(doc
))
865 return (wxDocument
*) NULL
;
868 // New document: user chooses a template, unless there's only one.
869 if (flags
& wxDOC_NEW
)
873 wxDocTemplate
*temp
= templates
[0];
875 wxDocument
*newDoc
= temp
->CreateDocument(path
, flags
);
878 newDoc
->SetDocumentName(temp
->GetDocumentName());
879 newDoc
->SetDocumentTemplate(temp
);
880 newDoc
->OnNewDocument();
885 wxDocTemplate
*temp
= SelectDocumentType(templates
, n
);
889 wxDocument
*newDoc
= temp
->CreateDocument(path
, flags
);
892 newDoc
->SetDocumentName(temp
->GetDocumentName());
893 newDoc
->SetDocumentTemplate(temp
);
894 newDoc
->OnNewDocument();
899 return (wxDocument
*) NULL
;
903 wxDocTemplate
*temp
= (wxDocTemplate
*) NULL
;
909 if (flags
& wxDOC_SILENT
)
910 temp
= FindTemplateForPath(path2
);
912 temp
= SelectDocumentPath(templates
, n
, path2
, flags
);
918 wxDocument
*newDoc
= temp
->CreateDocument(path2
, flags
);
921 newDoc
->SetDocumentName(temp
->GetDocumentName());
922 newDoc
->SetDocumentTemplate(temp
);
923 if (!newDoc
->OnOpenDocument(path2
))
926 return (wxDocument
*) NULL
;
928 AddFileToHistory(path2
);
933 return (wxDocument
*) NULL
;
936 wxView
*wxDocManager::CreateView(wxDocument
*doc
, long flags
)
938 wxDocTemplate
**templates
= new wxDocTemplate
*[m_templates
.Number()];
941 for (i
= 0; i
< m_templates
.Number(); i
++)
943 wxDocTemplate
*temp
= (wxDocTemplate
*)(m_templates
.Nth(i
)->Data());
944 if (temp
->IsVisible())
946 if (temp
->GetDocumentName() == doc
->GetDocumentName())
956 return (wxView
*) NULL
;
960 wxDocTemplate
*temp
= templates
[0];
962 wxView
*view
= temp
->CreateView(doc
, flags
);
964 view
->SetViewName(temp
->GetViewName());
968 wxDocTemplate
*temp
= SelectViewType(templates
, n
);
972 wxView
*view
= temp
->CreateView(doc
, flags
);
974 view
->SetViewName(temp
->GetViewName());
978 return (wxView
*) NULL
;
981 // Not yet implemented
982 void wxDocManager::DeleteTemplate(wxDocTemplate
*WXUNUSED(temp
), long WXUNUSED(flags
))
986 // Not yet implemented
987 bool wxDocManager::FlushDoc(wxDocument
*WXUNUSED(doc
))
992 wxDocument
*wxDocManager::GetCurrentDocument(void) const
995 return m_currentView
->GetDocument();
997 return (wxDocument
*) NULL
;
1000 // Make a default document name
1001 bool wxDocManager::MakeDefaultName(wxString
& name
)
1004 sprintf(buf
, _("unnamed%d"), m_defaultDocumentNameCounter
);
1005 m_defaultDocumentNameCounter
++;
1010 // Not yet implemented
1011 wxDocTemplate
*wxDocManager::MatchTemplate(const wxString
& WXUNUSED(path
))
1013 return (wxDocTemplate
*) NULL
;
1016 // File history management
1017 void wxDocManager::AddFileToHistory(const wxString
& file
)
1020 m_fileHistory
->AddFileToHistory(file
);
1023 wxString
wxDocManager::GetHistoryFile(int i
) const
1026 return wxString(m_fileHistory
->GetHistoryFile(i
));
1028 return wxString("");
1031 void wxDocManager::FileHistoryUseMenu(wxMenu
*menu
)
1034 m_fileHistory
->UseMenu(menu
);
1037 void wxDocManager::FileHistoryRemoveMenu(wxMenu
*menu
)
1040 m_fileHistory
->RemoveMenu(menu
);
1043 void wxDocManager::FileHistoryLoad(wxConfigBase
& config
)
1046 m_fileHistory
->Load(config
);
1049 void wxDocManager::FileHistorySave(wxConfigBase
& config
)
1052 m_fileHistory
->Save(config
);
1055 void wxDocManager::FileHistoryAddFilesToMenu(wxMenu
* menu
)
1058 m_fileHistory
->AddFilesToMenu(menu
);
1061 void wxDocManager::FileHistoryAddFilesToMenu()
1064 m_fileHistory
->AddFilesToMenu();
1067 int wxDocManager::GetNoHistoryFiles(void) const
1070 return m_fileHistory
->GetNoHistoryFiles();
1075 static char *FindExtension(char *path
)
1077 static char ext
[10];
1078 int len
= strlen(path
);
1082 for (i
= (len
-1); i
> 0; i
--)
1088 for (j
= i
+1; j
< len
; j
++)
1089 ext
[(int)(j
-(i
+1))] = (char)wxToLower(path
[j
]); // NOTE Should not use tolower under UNIX
1094 return (char *) NULL
;
1096 else return (char *) NULL
;
1100 // Given a path, try to find a matching template. Won't
1101 // always work, of course.
1102 wxDocTemplate
*wxDocManager::FindTemplateForPath(const wxString
& path
)
1104 char *theExt
= FindExtension((char *)(const char *)path
);
1106 return (wxDocTemplate
*) NULL
;
1107 wxDocTemplate
*theTemplate
= (wxDocTemplate
*) NULL
;
1109 if (m_templates
.Number() == 1)
1110 return (wxDocTemplate
*)m_templates
.First()->Data();
1112 // Find the template which this extension corresponds to
1114 for (i
= 0; i
< m_templates
.Number(); i
++)
1116 wxDocTemplate
*temp
= (wxDocTemplate
*)m_templates
.Nth(i
)->Data();
1117 if (strcmp(temp
->GetDefaultExtension(), theExt
) == 0)
1126 // Prompts user to open a file, using file specs in templates.
1127 // How to implement in wxWindows? Must extend the file selector
1128 // dialog or implement own; OR match the extension to the
1129 // template extension.
1130 wxDocTemplate
*wxDocManager::SelectDocumentPath(wxDocTemplate
**templates
,
1131 int noTemplates
, wxString
& path
, long WXUNUSED(flags
), bool WXUNUSED(save
))
1133 // We can only have multiple filters in Windows
1135 char *descrBuf
= new char[1000];
1138 for (i
= 0; i
< noTemplates
; i
++)
1140 if (templates
[i
]->IsVisible())
1142 strcat(descrBuf
, templates
[i
]->GetDescription());
1143 strcat(descrBuf
, " (");
1144 strcat(descrBuf
, templates
[i
]->GetFileFilter());
1145 strcat(descrBuf
, ") ");
1146 strcat(descrBuf
, "|");
1147 strcat(descrBuf
, templates
[i
]->GetFileFilter());
1148 strcat(descrBuf
, "|");
1151 int len
= strlen(descrBuf
);
1154 descrBuf
[len
-1] = 0;
1156 char *pathTmp
= wxFileSelector(_("Select a file"), "", "", "", descrBuf
, 0, wxTheApp
->GetTopWindow());
1161 char *theExt
= FindExtension((char *)(const char *)path
);
1163 return (wxDocTemplate
*) NULL
;
1165 // This is dodgy in that we're selecting the template on the
1166 // basis of the file extension, which may not be a standard
1167 // one. We really want to know exactly which template was
1168 // chosen by using a more advanced file selector.
1169 wxDocTemplate
*theTemplate
= FindTemplateForPath(path
);
1175 return (wxDocTemplate
*) NULL
;
1178 // In all other windowing systems, until we have more advanced
1179 // file selectors, we must select the document type (template) first, and
1180 // _then_ pop up the file selector.
1181 wxDocTemplate
*temp
= SelectDocumentType(templates
, noTemplates
);
1183 return (wxDocTemplate
*) NULL
;
1185 char *pathTmp
= wxFileSelector(_("Select a file"), "", "",
1186 temp
->GetDefaultExtension(),
1187 temp
->GetFileFilter(),
1188 0, wxTheApp
->GetTopWindow());
1196 return (wxDocTemplate
*) NULL
;
1200 wxDocTemplate
*wxDocManager::SelectDocumentType(wxDocTemplate
**templates
,
1203 char **strings
= new char *[noTemplates
];
1204 char **data
= new char *[noTemplates
];
1207 for (i
= 0; i
< noTemplates
; i
++)
1209 if (templates
[i
]->IsVisible())
1211 strings
[n
] = WXSTRINGCAST templates
[i
]->m_description
;
1212 data
[n
] = (char *)templates
[i
];
1220 return (wxDocTemplate
*) NULL
;
1224 wxDocTemplate
*temp
= (wxDocTemplate
*)data
[0];
1230 wxDocTemplate
*theTemplate
= (wxDocTemplate
*)wxGetSingleChoiceData(_("Select a document template"), _("Templates"), n
,
1237 wxDocTemplate
*wxDocManager::SelectViewType(wxDocTemplate
**templates
,
1240 char **strings
= new char *[noTemplates
];
1241 char **data
= new char *[noTemplates
];
1244 for (i
= 0; i
< noTemplates
; i
++)
1246 if (templates
[i
]->IsVisible() && templates
[i
]->GetViewName())
1248 strings
[n
] = WXSTRINGCAST templates
[i
]->m_viewTypeName
;
1249 data
[n
] = (char *)templates
[i
];
1253 wxDocTemplate
*theTemplate
= (wxDocTemplate
*)wxGetSingleChoiceData(_("Select a document view"), _("Views"), n
,
1260 void wxDocManager::AssociateTemplate(wxDocTemplate
*temp
)
1262 if (!m_templates
.Member(temp
))
1263 m_templates
.Append(temp
);
1266 void wxDocManager::DisassociateTemplate(wxDocTemplate
*temp
)
1268 m_templates
.DeleteObject(temp
);
1271 // Add and remove a document from the manager's list
1272 void wxDocManager::AddDocument(wxDocument
*doc
)
1274 if (!m_docs
.Member(doc
))
1278 void wxDocManager::RemoveDocument(wxDocument
*doc
)
1280 m_docs
.DeleteObject(doc
);
1283 // Views or windows should inform the document manager
1284 // when a view is going in or out of focus
1285 void wxDocManager::ActivateView(wxView
*view
, bool activate
, bool WXUNUSED(deleting
))
1287 // If we're deactiving, and if we're not actually deleting the view, then
1288 // don't reset the current view because we may be going to
1289 // a window without a view.
1290 // WHAT DID I MEAN BY THAT EXACTLY?
1294 if (m_currentView == view)
1295 m_currentView = NULL;
1301 m_currentView
= view
;
1303 m_currentView
= (wxView
*) NULL
;
1308 * Default document child frame
1311 BEGIN_EVENT_TABLE(wxDocChildFrame
, wxFrame
)
1312 EVT_ACTIVATE(wxDocChildFrame::OnActivate
)
1315 wxDocChildFrame::wxDocChildFrame(wxDocument
*doc
, wxView
*view
, wxFrame
*frame
, wxWindowID id
, const wxString
& title
,
1316 const wxPoint
& pos
, const wxSize
& size
, long style
, const wxString
& name
):
1317 wxFrame(frame
, id
, title
, pos
, size
, style
, name
)
1319 m_childDocument
= doc
;
1322 view
->SetFrame(this);
1325 wxDocChildFrame::~wxDocChildFrame(void)
1329 // Extend event processing to search the view's event table
1330 bool wxDocChildFrame::ProcessEvent(wxEvent
& event
)
1333 m_childView
->Activate(TRUE
);
1335 if ( !m_childView
|| ! m_childView
->ProcessEvent(event
) )
1337 // Only hand up to the parent if it's a menu command
1338 if (!event
.IsKindOf(CLASSINFO(wxCommandEvent
)) || !GetParent() || !GetParent()->ProcessEvent(event
))
1339 return wxEvtHandler::ProcessEvent(event
);
1347 void wxDocChildFrame::OnActivate(wxActivateEvent
& event
)
1349 wxFrame::OnActivate(event
);
1352 m_childView
->Activate(event
.GetActive());
1355 bool wxDocChildFrame::OnClose(void)
1357 // Close view but don't delete the frame while doing so!
1358 // ...since it will be deleted by wxWindows if we return TRUE.
1361 bool ans
= m_childView
->Close(FALSE
); // FALSE means don't delete associated window
1364 m_childView
->Activate(FALSE
);
1366 m_childView
= (wxView
*) NULL
;
1367 m_childDocument
= (wxDocument
*) NULL
;
1376 * Default parent frame
1379 BEGIN_EVENT_TABLE(wxDocParentFrame
, wxFrame
)
1380 EVT_MENU(wxID_EXIT
, wxDocParentFrame::OnExit
)
1381 EVT_MENU_RANGE(wxID_FILE1
, wxID_FILE9
, wxDocParentFrame::OnMRUFile
)
1384 wxDocParentFrame::wxDocParentFrame(wxDocManager
*manager
, wxFrame
*frame
, wxWindowID id
, const wxString
& title
,
1385 const wxPoint
& pos
, const wxSize
& size
, long style
, const wxString
& name
):
1386 wxFrame(frame
, id
, title
, pos
, size
, style
, name
)
1388 m_docManager
= manager
;
1391 void wxDocParentFrame::OnExit(wxCommandEvent
& WXUNUSED(event
))
1396 void wxDocParentFrame::OnMRUFile(wxCommandEvent
& event
)
1398 wxString
f(m_docManager
->GetHistoryFile(event
.GetSelection() - wxID_FILE1
));
1400 (void)m_docManager
->CreateDocument(f
, wxDOC_SILENT
);
1403 // Extend event processing to search the view's event table
1404 bool wxDocParentFrame::ProcessEvent(wxEvent
& event
)
1406 // Try the document manager, then do default processing
1407 if (!m_docManager
|| !m_docManager
->ProcessEvent(event
))
1408 return wxEvtHandler::ProcessEvent(event
);
1413 // Define the behaviour for the frame closing
1414 // - must delete all frames except for the main one.
1415 bool wxDocParentFrame::OnClose(void)
1417 return m_docManager
->Clear(FALSE
);
1420 #if wxUSE_PRINTING_ARCHITECTURE
1422 wxDocPrintout::wxDocPrintout(wxView
*view
, const wxString
& title
):
1423 wxPrintout(WXSTRINGCAST title
)
1425 m_printoutView
= view
;
1428 bool wxDocPrintout::OnPrintPage(int WXUNUSED(page
))
1432 // Get the logical pixels per inch of screen and printer
1433 int ppiScreenX
, ppiScreenY
;
1434 GetPPIScreen(&ppiScreenX
, &ppiScreenY
);
1435 int ppiPrinterX
, ppiPrinterY
;
1436 GetPPIPrinter(&ppiPrinterX
, &ppiPrinterY
);
1438 // This scales the DC so that the printout roughly represents the
1439 // the screen scaling. The text point size _should_ be the right size
1440 // but in fact is too small for some reason. This is a detail that will
1441 // need to be addressed at some point but can be fudged for the
1443 float scale
= (float)((float)ppiPrinterX
/(float)ppiScreenX
);
1445 // Now we have to check in case our real page size is reduced
1446 // (e.g. because we're drawing to a print preview memory DC)
1447 int pageWidth
, pageHeight
;
1449 dc
->GetSize(&w
, &h
);
1450 GetPageSizePixels(&pageWidth
, &pageHeight
);
1452 // If printer pageWidth == current DC width, then this doesn't
1453 // change. But w might be the preview bitmap width, so scale down.
1454 float overallScale
= scale
* (float)(w
/(float)pageWidth
);
1455 dc
->SetUserScale(overallScale
, overallScale
);
1459 m_printoutView
->OnDraw(dc
);
1464 bool wxDocPrintout::HasPage(int pageNum
)
1466 return (pageNum
== 1);
1469 bool wxDocPrintout::OnBeginDocument(int startPage
, int endPage
)
1471 if (!wxPrintout::OnBeginDocument(startPage
, endPage
))
1477 void wxDocPrintout::GetPageInfo(int *minPage
, int *maxPage
, int *selPageFrom
, int *selPageTo
)
1488 * Command processing framework
1491 wxCommand::wxCommand(bool canUndoIt
, const wxString
& name
)
1493 m_canUndo
= canUndoIt
;
1494 m_commandName
= name
;
1497 wxCommand::~wxCommand(void)
1501 // Command processor
1502 wxCommandProcessor::wxCommandProcessor(int maxCommands
)
1504 m_maxNoCommands
= maxCommands
;
1505 m_currentCommand
= (wxNode
*) NULL
;
1506 m_commandEditMenu
= (wxMenu
*) NULL
;
1509 wxCommandProcessor::~wxCommandProcessor(void)
1514 // Pass a command to the processor. The processor calls Do();
1515 // if successful, is appended to the command history unless
1516 // storeIt is FALSE.
1517 bool wxCommandProcessor::Submit(wxCommand
*command
, bool storeIt
)
1519 bool success
= command
->Do();
1520 if (success
&& storeIt
)
1522 if (m_commands
.Number() == m_maxNoCommands
)
1524 wxNode
*firstNode
= m_commands
.First();
1525 wxCommand
*firstCommand
= (wxCommand
*)firstNode
->Data();
1526 delete firstCommand
;
1530 // Correct a bug: we must chop off the current 'branch'
1531 // so that we're at the end of the command list.
1532 if (!m_currentCommand
)
1536 wxNode
*node
= m_currentCommand
->Next();
1539 wxNode
*next
= node
->Next();
1540 delete (wxCommand
*)node
->Data();
1546 m_commands
.Append(command
);
1547 m_currentCommand
= m_commands
.Last();
1553 bool wxCommandProcessor::Undo(void)
1555 if (m_currentCommand
)
1557 wxCommand
*command
= (wxCommand
*)m_currentCommand
->Data();
1558 if (command
->CanUndo())
1560 bool success
= command
->Undo();
1563 m_currentCommand
= m_currentCommand
->Previous();
1572 bool wxCommandProcessor::Redo(void)
1574 wxCommand
*redoCommand
= (wxCommand
*) NULL
;
1575 wxNode
*redoNode
= (wxNode
*) NULL
;
1576 if (m_currentCommand
&& m_currentCommand
->Next())
1578 redoCommand
= (wxCommand
*)m_currentCommand
->Next()->Data();
1579 redoNode
= m_currentCommand
->Next();
1583 if (m_commands
.Number() > 0)
1585 redoCommand
= (wxCommand
*)m_commands
.First()->Data();
1586 redoNode
= m_commands
.First();
1592 bool success
= redoCommand
->Do();
1595 m_currentCommand
= redoNode
;
1603 bool wxCommandProcessor::CanUndo(void) const
1605 if (m_currentCommand
)
1606 return ((wxCommand
*)m_currentCommand
->Data())->CanUndo();
1610 bool wxCommandProcessor::CanRedo(void) const
1612 if ((m_currentCommand
!= (wxNode
*) NULL
) && (m_currentCommand
->Next() == (wxNode
*) NULL
))
1615 if ((m_currentCommand
!= (wxNode
*) NULL
) && (m_currentCommand
->Next() != (wxNode
*) NULL
))
1618 if ((m_currentCommand
== (wxNode
*) NULL
) && (m_commands
.Number() > 0))
1624 void wxCommandProcessor::Initialize(void)
1626 m_currentCommand
= m_commands
.Last();
1630 void wxCommandProcessor::SetMenuStrings(void)
1632 if (m_commandEditMenu
)
1635 if (m_currentCommand
)
1637 wxCommand
*command
= (wxCommand
*)m_currentCommand
->Data();
1638 wxString
commandName(command
->GetName());
1639 if (commandName
== "") commandName
= _("Unnamed command");
1640 bool canUndo
= command
->CanUndo();
1642 buf
= wxString(_("&Undo ")) + commandName
;
1644 buf
= wxString(_("Can't &Undo ")) + commandName
;
1646 m_commandEditMenu
->SetLabel(wxID_UNDO
, buf
);
1647 m_commandEditMenu
->Enable(wxID_UNDO
, canUndo
);
1649 // We can redo, if we're not at the end of the history.
1650 if (m_currentCommand
->Next())
1652 wxCommand
*redoCommand
= (wxCommand
*)m_currentCommand
->Next()->Data();
1653 wxString
redoCommandName(redoCommand
->GetName());
1654 if (redoCommandName
== "") redoCommandName
= _("Unnamed command");
1655 buf
= wxString(_("&Redo ")) + redoCommandName
;
1656 m_commandEditMenu
->SetLabel(wxID_REDO
, buf
);
1657 m_commandEditMenu
->Enable(wxID_REDO
, TRUE
);
1661 m_commandEditMenu
->SetLabel(wxID_REDO
, _("&Redo"));
1662 m_commandEditMenu
->Enable(wxID_REDO
, FALSE
);
1667 m_commandEditMenu
->SetLabel(wxID_UNDO
, _("&Undo"));
1668 m_commandEditMenu
->Enable(wxID_UNDO
, FALSE
);
1670 if (m_commands
.Number() == 0)
1672 m_commandEditMenu
->SetLabel(wxID_REDO
, _("&Redo"));
1673 m_commandEditMenu
->Enable(wxID_REDO
, FALSE
);
1677 // currentCommand is NULL but there are commands: this means that
1678 // we've undone to the start of the list, but can redo the first.
1679 wxCommand
*redoCommand
= (wxCommand
*)m_commands
.First()->Data();
1680 wxString
redoCommandName(redoCommand
->GetName());
1681 if (!redoCommandName
) redoCommandName
= _("Unnamed command");
1682 buf
= wxString(_("&Redo ")) + redoCommandName
;
1683 m_commandEditMenu
->SetLabel(wxID_REDO
, buf
);
1684 m_commandEditMenu
->Enable(wxID_REDO
, TRUE
);
1690 void wxCommandProcessor::ClearCommands(void)
1692 wxNode
*node
= m_commands
.First();
1695 wxCommand
*command
= (wxCommand
*)node
->Data();
1698 node
= m_commands
.First();
1700 m_currentCommand
= (wxNode
*) NULL
;
1705 * File history processor
1708 wxFileHistory::wxFileHistory(int maxFiles
)
1710 m_fileMaxFiles
= maxFiles
;
1712 m_fileHistory
= new char *[m_fileMaxFiles
];
1715 wxFileHistory::~wxFileHistory(void)
1718 for (i
= 0; i
< m_fileHistoryN
; i
++)
1719 delete[] m_fileHistory
[i
];
1720 delete[] m_fileHistory
;
1723 // File history management
1724 void wxFileHistory::AddFileToHistory(const wxString
& file
)
1727 // Check we don't already have this file
1728 for (i
= 0; i
< m_fileHistoryN
; i
++)
1730 if (m_fileHistory
[i
] && wxString(m_fileHistory
[i
]) == file
)
1734 // Add to the project file history:
1735 // Move existing files (if any) down so we can insert file at beginning.
1737 // First delete filename that has popped off the end of the array (if any)
1738 if (m_fileHistoryN
== m_fileMaxFiles
)
1740 delete[] m_fileHistory
[m_fileMaxFiles
-1];
1741 m_fileHistory
[m_fileMaxFiles
-1] = (char *) NULL
;
1743 if (m_fileHistoryN
< m_fileMaxFiles
)
1745 wxNode
* node
= m_fileMenus
.First();
1748 wxMenu
* menu
= (wxMenu
*) node
->Data();
1749 if (m_fileHistoryN
== 0)
1750 menu
->AppendSeparator();
1751 menu
->Append(wxID_FILE1
+m_fileHistoryN
, _("[EMPTY]"));
1752 node
= node
->Next();
1756 // Shuffle filenames down
1757 for (i
= (m_fileHistoryN
-1); i
> 0; i
--)
1759 m_fileHistory
[i
] = m_fileHistory
[i
-1];
1761 m_fileHistory
[0] = copystring(file
);
1763 for (i
= 0; i
< m_fileHistoryN
; i
++)
1764 if (m_fileHistory
[i
])
1767 sprintf(buf
, "&%d %s", i
+1, m_fileHistory
[i
]);
1768 wxNode
* node
= m_fileMenus
.First();
1771 wxMenu
* menu
= (wxMenu
*) node
->Data();
1772 menu
->SetLabel(wxID_FILE1
+i
, buf
);
1773 node
= node
->Next();
1778 wxString
wxFileHistory::GetHistoryFile(int i
) const
1780 if (i
< m_fileHistoryN
)
1781 return wxString(m_fileHistory
[i
]);
1783 return wxString("");
1786 void wxFileHistory::UseMenu(wxMenu
*menu
)
1788 if (!m_fileMenus
.Member(menu
))
1789 m_fileMenus
.Append(menu
);
1792 void wxFileHistory::RemoveMenu(wxMenu
*menu
)
1794 m_fileMenus
.DeleteObject(menu
);
1797 void wxFileHistory::Load(wxConfigBase
& config
)
1801 sprintf(buf
, "file%d", m_fileHistoryN
+1);
1802 wxString
historyFile("");
1803 while ((m_fileHistoryN
<= m_fileMaxFiles
) && config
.Read(buf
, &historyFile
) && (historyFile
!= ""))
1805 m_fileHistory
[m_fileHistoryN
] = copystring((const char*) historyFile
);
1807 sprintf(buf
, "file%d", m_fileHistoryN
+1);
1813 void wxFileHistory::Save(wxConfigBase
& config
)
1816 for (i
= 0; i
< m_fileHistoryN
; i
++)
1819 buf
.Printf("file%d", i
+1);
1820 config
.Write(buf
, wxString(m_fileHistory
[i
]));
1824 void wxFileHistory::AddFilesToMenu()
1826 if (m_fileHistoryN
> 0)
1828 wxNode
* node
= m_fileMenus
.First();
1831 wxMenu
* menu
= (wxMenu
*) node
->Data();
1832 menu
->AppendSeparator();
1834 for (i
= 0; i
< m_fileHistoryN
; i
++)
1836 if (m_fileHistory
[i
])
1839 buf
.Printf("&%d %s", i
+1, m_fileHistory
[i
]);
1840 menu
->Append(wxID_FILE1
+i
, buf
);
1843 node
= node
->Next();
1848 void wxFileHistory::AddFilesToMenu(wxMenu
* menu
)
1850 if (m_fileHistoryN
> 0)
1852 menu
->AppendSeparator();
1854 for (i
= 0; i
< m_fileHistoryN
; i
++)
1856 if (m_fileHistory
[i
])
1859 buf
.Printf("&%d %s", i
+1, m_fileHistory
[i
]);
1860 menu
->Append(wxID_FILE1
+i
, buf
);
1871 wxPrintInfo::wxPrintInfo(void)
1876 wxPrintInfo::~wxPrintInfo(void)
1882 * Permits compatibility with existing file formats and functions
1883 * that manipulate files directly
1886 bool wxTransferFileToStream(const wxString
& filename
, ostream
& stream
)
1891 if ((fd1
= fopen (WXSTRINGCAST filename
, "rb")) == NULL
)
1894 while ((ch
= getc (fd1
)) != EOF
)
1895 stream
<< (unsigned char)ch
;
1901 bool wxTransferStreamToFile(istream
& stream
, const wxString
& filename
)
1906 if ((fd1
= fopen (WXSTRINGCAST filename
, "wb")) == NULL
)
1911 while (!stream
.eof())
1922 // End wxUSE_DOC_VIEW_ARCHITECTURE