]> git.saurik.com Git - wxWidgets.git/blame_incremental - src/common/docview.cpp
updates from Adrián González Alba
[wxWidgets.git] / src / common / docview.cpp
... / ...
CommitLineData
1/////////////////////////////////////////////////////////////////////////////
2// Name: src/common/docview.cpp
3// Purpose: Document/view classes
4// Author: Julian Smart
5// Modified by:
6// Created: 01/02/97
7// RCS-ID: $Id$
8// Copyright: (c) Julian Smart
9// Licence: wxWindows licence
10/////////////////////////////////////////////////////////////////////////////
11
12// ============================================================================
13// declarations
14// ============================================================================
15
16// ----------------------------------------------------------------------------
17// headers
18// ----------------------------------------------------------------------------
19
20// For compilers that support precompilation, includes "wx.h".
21#include "wx/wxprec.h"
22
23#ifdef __BORLANDC__
24 #pragma hdrstop
25#endif
26
27#if wxUSE_DOC_VIEW_ARCHITECTURE
28
29#include "wx/docview.h"
30
31#ifndef WX_PRECOMP
32 #include "wx/list.h"
33 #include "wx/string.h"
34 #include "wx/utils.h"
35 #include "wx/app.h"
36 #include "wx/dc.h"
37 #include "wx/dialog.h"
38 #include "wx/menu.h"
39 #include "wx/filedlg.h"
40 #include "wx/intl.h"
41 #include "wx/log.h"
42 #include "wx/msgdlg.h"
43 #include "wx/mdi.h"
44 #include "wx/choicdlg.h"
45#endif
46
47#include "wx/ffile.h"
48
49#ifdef __WXMAC__
50 #include "wx/filename.h"
51#endif
52
53#if wxUSE_PRINTING_ARCHITECTURE
54 #include "wx/prntbase.h"
55 #include "wx/printdlg.h"
56#endif
57
58#include "wx/confbase.h"
59#include "wx/file.h"
60#include "wx/cmdproc.h"
61#include "wx/tokenzr.h"
62
63#include <stdio.h>
64#include <string.h>
65
66#if wxUSE_STD_IOSTREAM
67 #include "wx/ioswrap.h"
68 #if wxUSE_IOSTREAMH
69 #include <fstream.h>
70 #else
71 #include <fstream>
72 #endif
73#else
74 #include "wx/wfstream.h"
75#endif
76
77// ----------------------------------------------------------------------------
78// wxWidgets macros
79// ----------------------------------------------------------------------------
80
81IMPLEMENT_ABSTRACT_CLASS(wxDocument, wxEvtHandler)
82IMPLEMENT_ABSTRACT_CLASS(wxView, wxEvtHandler)
83IMPLEMENT_ABSTRACT_CLASS(wxDocTemplate, wxObject)
84IMPLEMENT_DYNAMIC_CLASS(wxDocManager, wxEvtHandler)
85IMPLEMENT_CLASS(wxDocChildFrame, wxFrame)
86IMPLEMENT_CLASS(wxDocParentFrame, wxFrame)
87
88#if wxUSE_PRINTING_ARCHITECTURE
89 IMPLEMENT_DYNAMIC_CLASS(wxDocPrintout, wxPrintout)
90#endif
91
92IMPLEMENT_DYNAMIC_CLASS(wxFileHistory, wxObject)
93
94// ----------------------------------------------------------------------------
95// function prototypes
96// ----------------------------------------------------------------------------
97
98static inline wxString FindExtension(const wxChar *path);
99static wxWindow* wxFindSuitableParent(void);
100
101// ----------------------------------------------------------------------------
102// local constants
103// ----------------------------------------------------------------------------
104
105static const wxChar *s_MRUEntryFormat = wxT("&%d %s");
106
107// ============================================================================
108// implementation
109// ============================================================================
110
111// ----------------------------------------------------------------------------
112// local functions
113// ----------------------------------------------------------------------------
114
115static wxString FindExtension(const wxChar *path)
116{
117 wxString ext;
118 wxSplitPath(path, NULL, NULL, &ext);
119
120 // VZ: extensions are considered not case sensitive - is this really a good
121 // idea?
122 return ext.MakeLower();
123}
124
125// ----------------------------------------------------------------------------
126// Definition of wxDocument
127// ----------------------------------------------------------------------------
128
129wxDocument::wxDocument(wxDocument *parent)
130{
131 m_documentModified = false;
132 m_documentParent = parent;
133 m_documentTemplate = (wxDocTemplate *) NULL;
134 m_commandProcessor = (wxCommandProcessor*) NULL;
135 m_savedYet = false;
136}
137
138bool wxDocument::DeleteContents()
139{
140 return true;
141}
142
143wxDocument::~wxDocument()
144{
145 DeleteContents();
146
147 if (m_commandProcessor)
148 delete m_commandProcessor;
149
150 if (GetDocumentManager())
151 GetDocumentManager()->RemoveDocument(this);
152
153 // Not safe to do here, since it'll invoke virtual view functions
154 // expecting to see valid derived objects: and by the time we get here,
155 // we've called destructors higher up.
156 //DeleteAllViews();
157}
158
159bool wxDocument::Close()
160{
161 if (OnSaveModified())
162 return OnCloseDocument();
163 else
164 return false;
165}
166
167bool wxDocument::OnCloseDocument()
168{
169 // Tell all views that we're about to close
170 NotifyClosing();
171 DeleteContents();
172 Modify(false);
173 return true;
174}
175
176// Note that this implicitly deletes the document when the last view is
177// deleted.
178bool wxDocument::DeleteAllViews()
179{
180 wxDocManager* manager = GetDocumentManager();
181
182 // first check if all views agree to be closed
183 const wxList::iterator end = m_documentViews.end();
184 for ( wxList::iterator i = m_documentViews.begin(); i != end; ++i )
185 {
186 wxView *view = (wxView *)*i;
187 if ( !view->Close() )
188 return false;
189 }
190
191 // all views agreed to close, now do close them
192 if ( m_documentViews.empty() )
193 {
194 // normally the document would be implicitly deleted when the last view
195 // is, but if don't have any views, do it here instead
196 if ( manager && manager->GetDocuments().Member(this) )
197 delete this;
198 }
199 else // have views
200 {
201 // as we delete elements we iterate over, don't use the usual "from
202 // begin to end" loop
203 for ( ;; )
204 {
205 wxView *view = (wxView *)*m_documentViews.begin();
206
207 bool isLastOne = m_documentViews.size() == 1;
208
209 // this always deletes the node implicitly and if this is the last
210 // view also deletes this object itself (also implicitly, great),
211 // so we can't test for m_documentViews.empty() after calling this!
212 delete view;
213
214 if ( isLastOne )
215 break;
216 }
217 }
218
219 return true;
220}
221
222wxView *wxDocument::GetFirstView() const
223{
224 if (m_documentViews.GetCount() == 0)
225 return (wxView *) NULL;
226 return (wxView *)m_documentViews.GetFirst()->GetData();
227}
228
229wxDocManager *wxDocument::GetDocumentManager() const
230{
231 return (m_documentTemplate ? m_documentTemplate->GetDocumentManager() : (wxDocManager*) NULL);
232}
233
234bool wxDocument::OnNewDocument()
235{
236 if (!OnSaveModified())
237 return false;
238
239 if (OnCloseDocument()==false) return false;
240 DeleteContents();
241 Modify(false);
242 SetDocumentSaved(false);
243
244 wxString name;
245 GetDocumentManager()->MakeDefaultName(name);
246 SetTitle(name);
247 SetFilename(name, true);
248
249 return true;
250}
251
252bool wxDocument::Save()
253{
254 if (!IsModified() && m_savedYet)
255 return true;
256
257 if ( m_documentFile.empty() || !m_savedYet )
258 return SaveAs();
259
260 return OnSaveDocument(m_documentFile);
261}
262
263bool wxDocument::SaveAs()
264{
265 wxDocTemplate *docTemplate = GetDocumentTemplate();
266 if (!docTemplate)
267 return false;
268
269#if defined(__WXMSW__) || defined(__WXGTK__) || defined(__WXMAC__)
270 wxString filter = docTemplate->GetDescription() + wxT(" (") + docTemplate->GetFileFilter() + wxT(")|") + docTemplate->GetFileFilter();
271
272 // Now see if there are some other template with identical view and document
273 // classes, whose filters may also be used.
274
275 if (docTemplate->GetViewClassInfo() && docTemplate->GetDocClassInfo())
276 {
277 wxList::compatibility_iterator node = wxDocManager::GetDocumentManager()->GetTemplates().GetFirst();
278 while (node)
279 {
280 wxDocTemplate *t = (wxDocTemplate*) node->GetData();
281
282 if (t->IsVisible() && t != docTemplate &&
283 t->GetViewClassInfo() == docTemplate->GetViewClassInfo() &&
284 t->GetDocClassInfo() == docTemplate->GetDocClassInfo())
285 {
286 // add a '|' to separate this filter from the previous one
287 if ( !filter.empty() )
288 filter << wxT('|');
289
290 filter << t->GetDescription() << wxT(" (") << t->GetFileFilter() << wxT(") |")
291 << t->GetFileFilter();
292 }
293
294 node = node->GetNext();
295 }
296 }
297#else
298 wxString filter = docTemplate->GetFileFilter() ;
299#endif
300 wxString tmp = wxFileSelector(_("Save as"),
301 docTemplate->GetDirectory(),
302 wxFileNameFromPath(GetFilename()),
303 docTemplate->GetDefaultExtension(),
304 filter,
305 wxFD_SAVE | wxFD_OVERWRITE_PROMPT,
306 GetDocumentWindow());
307
308 if (tmp.empty())
309 return false;
310
311 wxString fileName(tmp);
312 wxString path, name, ext;
313 wxSplitPath(fileName, & path, & name, & ext);
314
315 if (ext.empty())
316 {
317 fileName += wxT(".");
318 fileName += docTemplate->GetDefaultExtension();
319 }
320
321 SetFilename(fileName);
322 SetTitle(wxFileNameFromPath(fileName));
323
324 // Notify the views that the filename has changed
325 wxList::compatibility_iterator node = m_documentViews.GetFirst();
326 while (node)
327 {
328 wxView *view = (wxView *)node->GetData();
329 view->OnChangeFilename();
330 node = node->GetNext();
331 }
332
333 // Files that were not saved correctly are not added to the FileHistory.
334 if (!OnSaveDocument(m_documentFile))
335 return false;
336
337 // A file that doesn't use the default extension of its document template cannot be opened
338 // via the FileHistory, so we do not add it.
339 if (docTemplate->FileMatchesTemplate(fileName))
340 {
341 GetDocumentManager()->AddFileToHistory(fileName);
342 }
343 else
344 {
345 // The user will probably not be able to open the file again, so
346 // we could warn about the wrong file-extension here.
347 }
348 return true;
349}
350
351bool wxDocument::OnSaveDocument(const wxString& file)
352{
353 if ( !file )
354 return false;
355
356 if ( !DoSaveDocument(file) )
357 return false;
358
359 Modify(false);
360 SetFilename(file);
361 SetDocumentSaved(true);
362#ifdef __WXMAC__
363 wxFileName fn(file) ;
364 fn.MacSetDefaultTypeAndCreator() ;
365#endif
366 return true;
367}
368
369bool wxDocument::OnOpenDocument(const wxString& file)
370{
371 if (!OnSaveModified())
372 return false;
373
374 if ( !DoOpenDocument(file) )
375 return false;
376
377 SetFilename(file, true);
378 Modify(false);
379 m_savedYet = true;
380
381 UpdateAllViews();
382
383 return true;
384}
385
386#if wxUSE_STD_IOSTREAM
387wxSTD istream& wxDocument::LoadObject(wxSTD istream& stream)
388#else
389wxInputStream& wxDocument::LoadObject(wxInputStream& stream)
390#endif
391{
392 return stream;
393}
394
395#if wxUSE_STD_IOSTREAM
396wxSTD ostream& wxDocument::SaveObject(wxSTD ostream& stream)
397#else
398wxOutputStream& wxDocument::SaveObject(wxOutputStream& stream)
399#endif
400{
401 return stream;
402}
403
404bool wxDocument::Revert()
405{
406 return false;
407}
408
409
410// Get title, or filename if no title, else unnamed
411bool wxDocument::GetPrintableName(wxString& buf) const
412{
413 if (!m_documentTitle.empty())
414 {
415 buf = m_documentTitle;
416 return true;
417 }
418 else if (!m_documentFile.empty())
419 {
420 buf = wxFileNameFromPath(m_documentFile);
421 return true;
422 }
423 else
424 {
425 buf = _("unnamed");
426 return true;
427 }
428}
429
430wxWindow *wxDocument::GetDocumentWindow() const
431{
432 wxView *view = GetFirstView();
433 if (view)
434 return view->GetFrame();
435 else
436 return wxTheApp->GetTopWindow();
437}
438
439wxCommandProcessor *wxDocument::OnCreateCommandProcessor()
440{
441 return new wxCommandProcessor;
442}
443
444// true if safe to close
445bool wxDocument::OnSaveModified()
446{
447 if (IsModified())
448 {
449 wxString title;
450 GetPrintableName(title);
451
452 wxString msgTitle;
453 if (!wxTheApp->GetAppName().empty())
454 msgTitle = wxTheApp->GetAppName();
455 else
456 msgTitle = wxString(_("Warning"));
457
458 wxString prompt;
459 prompt.Printf(_("Do you want to save changes to document %s?"),
460 (const wxChar *)title);
461 int res = wxMessageBox(prompt, msgTitle,
462 wxYES_NO|wxCANCEL|wxICON_QUESTION,
463 GetDocumentWindow());
464 if (res == wxNO)
465 {
466 Modify(false);
467 return true;
468 }
469 else if (res == wxYES)
470 return Save();
471 else if (res == wxCANCEL)
472 return false;
473 }
474 return true;
475}
476
477bool wxDocument::Draw(wxDC& WXUNUSED(context))
478{
479 return true;
480}
481
482bool wxDocument::AddView(wxView *view)
483{
484 if (!m_documentViews.Member(view))
485 {
486 m_documentViews.Append(view);
487 OnChangedViewList();
488 }
489 return true;
490}
491
492bool wxDocument::RemoveView(wxView *view)
493{
494 (void)m_documentViews.DeleteObject(view);
495 OnChangedViewList();
496 return true;
497}
498
499bool wxDocument::OnCreate(const wxString& WXUNUSED(path), long flags)
500{
501 if (GetDocumentTemplate()->CreateView(this, flags))
502 return true;
503 else
504 return false;
505}
506
507// Called after a view is added or removed.
508// The default implementation deletes the document if
509// there are no more views.
510void wxDocument::OnChangedViewList()
511{
512 if (m_documentViews.GetCount() == 0)
513 {
514 if (OnSaveModified())
515 {
516 delete this;
517 }
518 }
519}
520
521void wxDocument::UpdateAllViews(wxView *sender, wxObject *hint)
522{
523 wxList::compatibility_iterator node = m_documentViews.GetFirst();
524 while (node)
525 {
526 wxView *view = (wxView *)node->GetData();
527 if (view != sender)
528 view->OnUpdate(sender, hint);
529 node = node->GetNext();
530 }
531}
532
533void wxDocument::NotifyClosing()
534{
535 wxList::compatibility_iterator node = m_documentViews.GetFirst();
536 while (node)
537 {
538 wxView *view = (wxView *)node->GetData();
539 view->OnClosingDocument();
540 node = node->GetNext();
541 }
542}
543
544void wxDocument::SetFilename(const wxString& filename, bool notifyViews)
545{
546 m_documentFile = filename;
547 if ( notifyViews )
548 {
549 // Notify the views that the filename has changed
550 wxList::compatibility_iterator node = m_documentViews.GetFirst();
551 while (node)
552 {
553 wxView *view = (wxView *)node->GetData();
554 view->OnChangeFilename();
555 node = node->GetNext();
556 }
557 }
558}
559
560bool wxDocument::DoSaveDocument(const wxString& file)
561{
562 wxString msgTitle;
563 if (!wxTheApp->GetAppName().empty())
564 msgTitle = wxTheApp->GetAppName();
565 else
566 msgTitle = wxString(_("File error"));
567
568#if wxUSE_STD_IOSTREAM
569 wxSTD ofstream store(file.mb_str(), wxSTD ios::binary);
570 if (store.fail() || store.bad())
571#else
572 wxFileOutputStream store(file);
573 if (store.GetLastError() != wxSTREAM_NO_ERROR)
574#endif
575 {
576 (void)wxMessageBox(_("Sorry, could not open this file for saving."), msgTitle, wxOK | wxICON_EXCLAMATION,
577 GetDocumentWindow());
578 // Saving error
579 return false;
580 }
581 if (!SaveObject(store))
582 {
583 (void)wxMessageBox(_("Sorry, could not save this file."), msgTitle, wxOK | wxICON_EXCLAMATION,
584 GetDocumentWindow());
585 // Saving error
586 return false;
587 }
588
589 return true;
590}
591
592bool wxDocument::DoOpenDocument(const wxString& file)
593{
594#if wxUSE_STD_IOSTREAM
595 wxSTD ifstream store(file.mb_str(), wxSTD ios::binary);
596 if (!store.fail() && !store.bad())
597#else
598 wxFileInputStream store(file);
599 if (store.GetLastError() == wxSTREAM_NO_ERROR)
600#endif
601 {
602#if wxUSE_STD_IOSTREAM
603 LoadObject(store);
604 if ( !!store || store.eof() )
605#else
606 int res = LoadObject(store).GetLastError();
607 if ( res == wxSTREAM_NO_ERROR || res == wxSTREAM_EOF )
608#endif
609 return true;
610 }
611
612 wxLogError(_("Sorry, could not open this file."));
613 return false;
614}
615
616
617// ----------------------------------------------------------------------------
618// Document view
619// ----------------------------------------------------------------------------
620
621wxView::wxView()
622{
623 m_viewDocument = (wxDocument*) NULL;
624
625 m_viewFrame = (wxFrame *) NULL;
626}
627
628wxView::~wxView()
629{
630 GetDocumentManager()->ActivateView(this, false);
631 m_viewDocument->RemoveView(this);
632}
633
634// Extend event processing to search the document's event table
635bool wxView::ProcessEvent(wxEvent& event)
636{
637 if ( !GetDocument() || !GetDocument()->ProcessEvent(event) )
638 return wxEvtHandler::ProcessEvent(event);
639
640 return true;
641}
642
643void wxView::OnActivateView(bool WXUNUSED(activate), wxView *WXUNUSED(activeView), wxView *WXUNUSED(deactiveView))
644{
645}
646
647void wxView::OnPrint(wxDC *dc, wxObject *WXUNUSED(info))
648{
649 OnDraw(dc);
650}
651
652void wxView::OnUpdate(wxView *WXUNUSED(sender), wxObject *WXUNUSED(hint))
653{
654}
655
656void wxView::OnChangeFilename()
657{
658 // GetFrame can return wxWindow rather than wxTopLevelWindow due to
659 // generic MDI implementation so use SetLabel rather than SetTitle.
660 // It should cause SetTitle() for top level windows.
661 wxWindow *win = GetFrame();
662 if (!win) return;
663
664 wxDocument *doc = GetDocument();
665 if (!doc) return;
666
667 wxString name;
668 doc->GetPrintableName(name);
669 win->SetLabel(name);
670}
671
672void wxView::SetDocument(wxDocument *doc)
673{
674 m_viewDocument = doc;
675 if (doc)
676 doc->AddView(this);
677}
678
679bool wxView::Close(bool deleteWindow)
680{
681 if (OnClose(deleteWindow))
682 return true;
683 else
684 return false;
685}
686
687void wxView::Activate(bool activate)
688{
689 if (GetDocument() && GetDocumentManager())
690 {
691 OnActivateView(activate, this, GetDocumentManager()->GetCurrentView());
692 GetDocumentManager()->ActivateView(this, activate);
693 }
694}
695
696bool wxView::OnClose(bool WXUNUSED(deleteWindow))
697{
698 return GetDocument() ? GetDocument()->Close() : true;
699}
700
701#if wxUSE_PRINTING_ARCHITECTURE
702wxPrintout *wxView::OnCreatePrintout()
703{
704 return new wxDocPrintout(this);
705}
706#endif // wxUSE_PRINTING_ARCHITECTURE
707
708// ----------------------------------------------------------------------------
709// wxDocTemplate
710// ----------------------------------------------------------------------------
711
712wxDocTemplate::wxDocTemplate(wxDocManager *manager,
713 const wxString& descr,
714 const wxString& filter,
715 const wxString& dir,
716 const wxString& ext,
717 const wxString& docTypeName,
718 const wxString& viewTypeName,
719 wxClassInfo *docClassInfo,
720 wxClassInfo *viewClassInfo,
721 long flags)
722{
723 m_documentManager = manager;
724 m_description = descr;
725 m_directory = dir;
726 m_defaultExt = ext;
727 m_fileFilter = filter;
728 m_flags = flags;
729 m_docTypeName = docTypeName;
730 m_viewTypeName = viewTypeName;
731 m_documentManager->AssociateTemplate(this);
732
733 m_docClassInfo = docClassInfo;
734 m_viewClassInfo = viewClassInfo;
735}
736
737wxDocTemplate::~wxDocTemplate()
738{
739 m_documentManager->DisassociateTemplate(this);
740}
741
742// Tries to dynamically construct an object of the right class.
743wxDocument *wxDocTemplate::CreateDocument(const wxString& path, long flags)
744{
745 wxDocument *doc = DoCreateDocument();
746 if ( doc == NULL )
747 return (wxDocument *) NULL;
748
749 if (InitDocument(doc, path, flags))
750 {
751 return doc;
752 }
753 else
754 {
755 return (wxDocument *) NULL;
756 }
757}
758
759bool wxDocTemplate::InitDocument(wxDocument* doc, const wxString& path, long flags)
760{
761 doc->SetFilename(path);
762 doc->SetDocumentTemplate(this);
763 GetDocumentManager()->AddDocument(doc);
764 doc->SetCommandProcessor(doc->OnCreateCommandProcessor());
765
766 if (doc->OnCreate(path, flags))
767 return true;
768 else
769 {
770 if (GetDocumentManager()->GetDocuments().Member(doc))
771 doc->DeleteAllViews();
772 return false;
773 }
774}
775
776wxView *wxDocTemplate::CreateView(wxDocument *doc, long flags)
777{
778 wxView *view = DoCreateView();
779 if ( view == NULL )
780 return (wxView *) NULL;
781
782 view->SetDocument(doc);
783 if (view->OnCreate(doc, flags))
784 {
785 return view;
786 }
787 else
788 {
789 delete view;
790 return (wxView *) NULL;
791 }
792}
793
794// The default (very primitive) format detection: check is the extension is
795// that of the template
796bool wxDocTemplate::FileMatchesTemplate(const wxString& path)
797{
798 wxStringTokenizer parser (GetFileFilter(), wxT(";"));
799 wxString anything = wxT ("*");
800 while (parser.HasMoreTokens())
801 {
802 wxString filter = parser.GetNextToken();
803 wxString filterExt = FindExtension (filter);
804 if ( filter.IsSameAs (anything) ||
805 filterExt.IsSameAs (anything) ||
806 filterExt.IsSameAs (FindExtension (path)) )
807 return true;
808 }
809 return GetDefaultExtension().IsSameAs(FindExtension(path));
810}
811
812wxDocument *wxDocTemplate::DoCreateDocument()
813{
814 if (!m_docClassInfo)
815 return (wxDocument *) NULL;
816
817 return (wxDocument *)m_docClassInfo->CreateObject();
818}
819
820wxView *wxDocTemplate::DoCreateView()
821{
822 if (!m_viewClassInfo)
823 return (wxView *) NULL;
824
825 return (wxView *)m_viewClassInfo->CreateObject();
826}
827
828// ----------------------------------------------------------------------------
829// wxDocManager
830// ----------------------------------------------------------------------------
831
832BEGIN_EVENT_TABLE(wxDocManager, wxEvtHandler)
833 EVT_MENU(wxID_OPEN, wxDocManager::OnFileOpen)
834 EVT_MENU(wxID_CLOSE, wxDocManager::OnFileClose)
835 EVT_MENU(wxID_CLOSE_ALL, wxDocManager::OnFileCloseAll)
836 EVT_MENU(wxID_REVERT, wxDocManager::OnFileRevert)
837 EVT_MENU(wxID_NEW, wxDocManager::OnFileNew)
838 EVT_MENU(wxID_SAVE, wxDocManager::OnFileSave)
839 EVT_MENU(wxID_SAVEAS, wxDocManager::OnFileSaveAs)
840 EVT_MENU(wxID_UNDO, wxDocManager::OnUndo)
841 EVT_MENU(wxID_REDO, wxDocManager::OnRedo)
842
843 EVT_UPDATE_UI(wxID_OPEN, wxDocManager::OnUpdateFileOpen)
844 EVT_UPDATE_UI(wxID_CLOSE, wxDocManager::OnUpdateFileClose)
845 EVT_UPDATE_UI(wxID_CLOSE_ALL, wxDocManager::OnUpdateFileClose)
846 EVT_UPDATE_UI(wxID_REVERT, wxDocManager::OnUpdateFileRevert)
847 EVT_UPDATE_UI(wxID_NEW, wxDocManager::OnUpdateFileNew)
848 EVT_UPDATE_UI(wxID_SAVE, wxDocManager::OnUpdateFileSave)
849 EVT_UPDATE_UI(wxID_SAVEAS, wxDocManager::OnUpdateFileSaveAs)
850 EVT_UPDATE_UI(wxID_UNDO, wxDocManager::OnUpdateUndo)
851 EVT_UPDATE_UI(wxID_REDO, wxDocManager::OnUpdateRedo)
852
853#if wxUSE_PRINTING_ARCHITECTURE
854 EVT_MENU(wxID_PRINT, wxDocManager::OnPrint)
855 EVT_MENU(wxID_PREVIEW, wxDocManager::OnPreview)
856
857 EVT_UPDATE_UI(wxID_PRINT, wxDocManager::OnUpdatePrint)
858 EVT_UPDATE_UI(wxID_PREVIEW, wxDocManager::OnUpdatePreview)
859#endif
860END_EVENT_TABLE()
861
862wxDocManager* wxDocManager::sm_docManager = (wxDocManager*) NULL;
863
864wxDocManager::wxDocManager(long flags, bool initialize)
865{
866 m_defaultDocumentNameCounter = 1;
867 m_flags = flags;
868 m_currentView = (wxView *) NULL;
869 m_maxDocsOpen = 10000;
870 m_fileHistory = (wxFileHistory *) NULL;
871 if (initialize)
872 Initialize();
873 sm_docManager = this;
874}
875
876wxDocManager::~wxDocManager()
877{
878 Clear();
879 if (m_fileHistory)
880 delete m_fileHistory;
881 sm_docManager = (wxDocManager*) NULL;
882}
883
884// closes the specified document
885bool wxDocManager::CloseDocument(wxDocument* doc, bool force)
886{
887 if (doc->Close() || force)
888 {
889 // Implicitly deletes the document when
890 // the last view is deleted
891 doc->DeleteAllViews();
892
893 // Check we're really deleted
894 if (m_docs.Member(doc))
895 delete doc;
896
897 return true;
898 }
899 return false;
900}
901
902bool wxDocManager::CloseDocuments(bool force)
903{
904 wxList::compatibility_iterator node = m_docs.GetFirst();
905 while (node)
906 {
907 wxDocument *doc = (wxDocument *)node->GetData();
908 wxList::compatibility_iterator next = node->GetNext();
909
910 if (!CloseDocument(doc, force))
911 return false;
912
913 // This assumes that documents are not connected in
914 // any way, i.e. deleting one document does NOT
915 // delete another.
916 node = next;
917 }
918 return true;
919}
920
921bool wxDocManager::Clear(bool force)
922{
923 if (!CloseDocuments(force))
924 return false;
925
926 m_currentView = NULL;
927
928 wxList::compatibility_iterator node = m_templates.GetFirst();
929 while (node)
930 {
931 wxDocTemplate *templ = (wxDocTemplate*) node->GetData();
932 wxList::compatibility_iterator next = node->GetNext();
933 delete templ;
934 node = next;
935 }
936 return true;
937}
938
939bool wxDocManager::Initialize()
940{
941 m_fileHistory = OnCreateFileHistory();
942 return true;
943}
944
945wxFileHistory *wxDocManager::OnCreateFileHistory()
946{
947 return new wxFileHistory;
948}
949
950void wxDocManager::OnFileClose(wxCommandEvent& WXUNUSED(event))
951{
952 wxDocument *doc = GetCurrentDocument();
953 if (!doc)
954 return;
955 if (doc->Close())
956 {
957 doc->DeleteAllViews();
958 if (m_docs.Member(doc))
959 delete doc;
960 }
961}
962
963void wxDocManager::OnFileCloseAll(wxCommandEvent& WXUNUSED(event))
964{
965 CloseDocuments(false);
966}
967
968void wxDocManager::OnFileNew(wxCommandEvent& WXUNUSED(event))
969{
970 CreateDocument( wxEmptyString, wxDOC_NEW );
971}
972
973void wxDocManager::OnFileOpen(wxCommandEvent& WXUNUSED(event))
974{
975 if ( !CreateDocument( wxEmptyString, 0) )
976 {
977 OnOpenFileFailure();
978 }
979}
980
981void wxDocManager::OnFileRevert(wxCommandEvent& WXUNUSED(event))
982{
983 wxDocument *doc = GetCurrentDocument();
984 if (!doc)
985 return;
986 doc->Revert();
987}
988
989void wxDocManager::OnFileSave(wxCommandEvent& WXUNUSED(event))
990{
991 wxDocument *doc = GetCurrentDocument();
992 if (!doc)
993 return;
994 doc->Save();
995}
996
997void wxDocManager::OnFileSaveAs(wxCommandEvent& WXUNUSED(event))
998{
999 wxDocument *doc = GetCurrentDocument();
1000 if (!doc)
1001 return;
1002 doc->SaveAs();
1003}
1004
1005void wxDocManager::OnPrint(wxCommandEvent& WXUNUSED(event))
1006{
1007#if wxUSE_PRINTING_ARCHITECTURE
1008 wxView *view = GetCurrentView();
1009 if (!view)
1010 return;
1011
1012 wxPrintout *printout = view->OnCreatePrintout();
1013 if (printout)
1014 {
1015 wxPrinter printer;
1016 printer.Print(view->GetFrame(), printout, true);
1017
1018 delete printout;
1019 }
1020#endif // wxUSE_PRINTING_ARCHITECTURE
1021}
1022
1023void wxDocManager::OnPreview(wxCommandEvent& WXUNUSED(event))
1024{
1025#if wxUSE_PRINTING_ARCHITECTURE
1026 wxView *view = GetCurrentView();
1027 if (!view)
1028 return;
1029
1030 wxPrintout *printout = view->OnCreatePrintout();
1031 if (printout)
1032 {
1033 // Pass two printout objects: for preview, and possible printing.
1034 wxPrintPreviewBase *preview = new wxPrintPreview(printout, view->OnCreatePrintout());
1035 if ( !preview->Ok() )
1036 {
1037 delete preview;
1038 wxMessageBox( _("Sorry, print preview needs a printer to be installed.") );
1039 return;
1040 }
1041
1042 wxPreviewFrame *frame = new wxPreviewFrame(preview, (wxFrame *)wxTheApp->GetTopWindow(), _("Print Preview"),
1043 wxPoint(100, 100), wxSize(600, 650));
1044 frame->Centre(wxBOTH);
1045 frame->Initialize();
1046 frame->Show(true);
1047 }
1048#endif // wxUSE_PRINTING_ARCHITECTURE
1049}
1050
1051void wxDocManager::OnUndo(wxCommandEvent& event)
1052{
1053 wxDocument *doc = GetCurrentDocument();
1054 if (!doc)
1055 return;
1056 if (doc->GetCommandProcessor())
1057 doc->GetCommandProcessor()->Undo();
1058 else
1059 event.Skip();
1060}
1061
1062void wxDocManager::OnRedo(wxCommandEvent& event)
1063{
1064 wxDocument *doc = GetCurrentDocument();
1065 if (!doc)
1066 return;
1067 if (doc->GetCommandProcessor())
1068 doc->GetCommandProcessor()->Redo();
1069 else
1070 event.Skip();
1071}
1072
1073// Handlers for UI update commands
1074
1075void wxDocManager::OnUpdateFileOpen(wxUpdateUIEvent& event)
1076{
1077 event.Enable( true );
1078}
1079
1080void wxDocManager::OnUpdateFileClose(wxUpdateUIEvent& event)
1081{
1082 wxDocument *doc = GetCurrentDocument();
1083 event.Enable( (doc != (wxDocument*) NULL) );
1084}
1085
1086void wxDocManager::OnUpdateFileRevert(wxUpdateUIEvent& event)
1087{
1088 wxDocument *doc = GetCurrentDocument();
1089 event.Enable( (doc != (wxDocument*) NULL) );
1090}
1091
1092void wxDocManager::OnUpdateFileNew(wxUpdateUIEvent& event)
1093{
1094 event.Enable( true );
1095}
1096
1097void wxDocManager::OnUpdateFileSave(wxUpdateUIEvent& event)
1098{
1099 wxDocument *doc = GetCurrentDocument();
1100 event.Enable( doc && doc->IsModified() );
1101}
1102
1103void wxDocManager::OnUpdateFileSaveAs(wxUpdateUIEvent& event)
1104{
1105 wxDocument *doc = GetCurrentDocument();
1106 event.Enable( (doc != (wxDocument*) NULL) );
1107}
1108
1109void wxDocManager::OnUpdateUndo(wxUpdateUIEvent& event)
1110{
1111 wxDocument *doc = GetCurrentDocument();
1112 if (!doc)
1113 event.Enable(false);
1114 else if (!doc->GetCommandProcessor())
1115 event.Skip();
1116 else
1117 {
1118 event.Enable( doc->GetCommandProcessor()->CanUndo() );
1119 doc->GetCommandProcessor()->SetMenuStrings();
1120 }
1121}
1122
1123void wxDocManager::OnUpdateRedo(wxUpdateUIEvent& event)
1124{
1125 wxDocument *doc = GetCurrentDocument();
1126 if (!doc)
1127 event.Enable(false);
1128 else if (!doc->GetCommandProcessor())
1129 event.Skip();
1130 else
1131 {
1132 event.Enable( doc->GetCommandProcessor()->CanRedo() );
1133 doc->GetCommandProcessor()->SetMenuStrings();
1134 }
1135}
1136
1137void wxDocManager::OnUpdatePrint(wxUpdateUIEvent& event)
1138{
1139 wxDocument *doc = GetCurrentDocument();
1140 event.Enable( (doc != (wxDocument*) NULL) );
1141}
1142
1143void wxDocManager::OnUpdatePreview(wxUpdateUIEvent& event)
1144{
1145 wxDocument *doc = GetCurrentDocument();
1146 event.Enable( (doc != (wxDocument*) NULL) );
1147}
1148
1149wxView *wxDocManager::GetCurrentView() const
1150{
1151 if (m_currentView)
1152 return m_currentView;
1153 if (m_docs.GetCount() == 1)
1154 {
1155 wxDocument* doc = (wxDocument*) m_docs.GetFirst()->GetData();
1156 return doc->GetFirstView();
1157 }
1158 return (wxView *) NULL;
1159}
1160
1161// Extend event processing to search the view's event table
1162bool wxDocManager::ProcessEvent(wxEvent& event)
1163{
1164 wxView* view = GetCurrentView();
1165 if (view)
1166 {
1167 if (view->ProcessEvent(event))
1168 return true;
1169 }
1170 return wxEvtHandler::ProcessEvent(event);
1171}
1172
1173wxDocument *wxDocManager::CreateDocument(const wxString& path, long flags)
1174{
1175 wxDocTemplate **templates = new wxDocTemplate *[m_templates.GetCount()];
1176 int n = 0;
1177
1178 for (size_t i = 0; i < m_templates.GetCount(); i++)
1179 {
1180 wxDocTemplate *temp = (wxDocTemplate *)(m_templates.Item(i)->GetData());
1181 if (temp->IsVisible())
1182 {
1183 templates[n] = temp;
1184 n ++;
1185 }
1186 }
1187 if (n == 0)
1188 {
1189 delete[] templates;
1190 return (wxDocument *) NULL;
1191 }
1192
1193 wxDocument* docToClose = NULL;
1194
1195 // If we've reached the max number of docs, close the
1196 // first one.
1197 if ( (int)GetDocuments().GetCount() >= m_maxDocsOpen )
1198 {
1199 wxDocument *doc = (wxDocument *)GetDocuments().GetFirst()->GetData();
1200 docToClose = doc;
1201 }
1202
1203 // New document: user chooses a template, unless there's only one.
1204 if (flags & wxDOC_NEW)
1205 {
1206 if (n == 1)
1207 {
1208 if (docToClose)
1209 {
1210 if (!CloseDocument(docToClose, false))
1211 {
1212 delete[] templates;
1213 return NULL;
1214 }
1215 }
1216
1217 wxDocTemplate *temp = templates[0];
1218 delete[] templates;
1219 wxDocument *newDoc = temp->CreateDocument(path, flags);
1220
1221 if (newDoc)
1222 {
1223 newDoc->SetDocumentName(temp->GetDocumentName());
1224 newDoc->SetDocumentTemplate(temp);
1225 if (!newDoc->OnNewDocument() )
1226 {
1227 // Document is implicitly deleted by DeleteAllViews
1228 newDoc->DeleteAllViews();
1229 return NULL;
1230 }
1231 }
1232 return newDoc;
1233 }
1234
1235 wxDocTemplate *temp = SelectDocumentType(templates, n);
1236 delete[] templates;
1237 if (temp)
1238 {
1239 if (docToClose)
1240 {
1241 if (!CloseDocument(docToClose, false))
1242 {
1243 return NULL;
1244 }
1245 }
1246
1247 wxDocument *newDoc = temp->CreateDocument(path, flags);
1248
1249 if (newDoc)
1250 {
1251 newDoc->SetDocumentName(temp->GetDocumentName());
1252 newDoc->SetDocumentTemplate(temp);
1253 if (!newDoc->OnNewDocument() )
1254 {
1255 // Document is implicitly deleted by DeleteAllViews
1256 newDoc->DeleteAllViews();
1257 return NULL;
1258 }
1259 }
1260 return newDoc;
1261 }
1262 else
1263 return (wxDocument *) NULL;
1264 }
1265
1266 // Existing document
1267 wxDocTemplate *temp;
1268
1269 wxString path2 = path;
1270
1271 if (flags & wxDOC_SILENT)
1272 {
1273 temp = FindTemplateForPath(path2);
1274 if (!temp)
1275 {
1276 // Since we do not add files with non-default extensions to the FileHistory this
1277 // can only happen if the application changes the allowed templates in runtime.
1278 (void)wxMessageBox(_("Sorry, the format for this file is unknown."),
1279 _("Open File"),
1280 wxOK | wxICON_EXCLAMATION, wxFindSuitableParent());
1281 }
1282 }
1283 else
1284 temp = SelectDocumentPath(templates, n, path2, flags);
1285
1286 delete[] templates;
1287
1288 if (temp)
1289 {
1290 if (docToClose)
1291 {
1292 if (!CloseDocument(docToClose, false))
1293 {
1294 return NULL;
1295 }
1296 }
1297
1298 //see if this file is already open
1299 for (size_t i = 0; i < GetDocuments().GetCount(); ++i)
1300 {
1301 wxDocument* currentDoc = (wxDocument*)(GetDocuments().Item(i)->GetData());
1302#ifdef __WXMSW__
1303 //file paths are case-insensitive on Windows
1304 if (path2.CmpNoCase(currentDoc->GetFilename()) == 0)
1305#else
1306 if (path2.Cmp(currentDoc->GetFilename()) == 0)
1307#endif
1308 {
1309 //file already open. Just activate it and return
1310 if (currentDoc->GetFirstView())
1311 {
1312 ActivateView(currentDoc->GetFirstView(), true);
1313 if (currentDoc->GetDocumentWindow())
1314 currentDoc->GetDocumentWindow()->SetFocus();
1315 return currentDoc;
1316 }
1317 }
1318 }
1319
1320 wxDocument *newDoc = temp->CreateDocument(path2, flags);
1321 if (newDoc)
1322 {
1323 newDoc->SetDocumentName(temp->GetDocumentName());
1324 newDoc->SetDocumentTemplate(temp);
1325 if (!newDoc->OnOpenDocument(path2))
1326 {
1327 newDoc->DeleteAllViews();
1328 // delete newDoc; // Implicitly deleted by DeleteAllViews
1329 return (wxDocument *) NULL;
1330 }
1331 // A file that doesn't use the default extension of its document
1332 // template cannot be opened via the FileHistory, so we do not
1333 // add it.
1334 if (temp->FileMatchesTemplate(path2))
1335 AddFileToHistory(path2);
1336 }
1337 return newDoc;
1338 }
1339
1340 return (wxDocument *) NULL;
1341}
1342
1343wxView *wxDocManager::CreateView(wxDocument *doc, long flags)
1344{
1345 wxDocTemplate **templates = new wxDocTemplate *[m_templates.GetCount()];
1346 int n =0;
1347
1348 for (size_t i = 0; i < m_templates.GetCount(); i++)
1349 {
1350 wxDocTemplate *temp = (wxDocTemplate *)(m_templates.Item(i)->GetData());
1351 if (temp->IsVisible())
1352 {
1353 if (temp->GetDocumentName() == doc->GetDocumentName())
1354 {
1355 templates[n] = temp;
1356 n ++;
1357 }
1358 }
1359 }
1360 if (n == 0)
1361 {
1362 delete[] templates;
1363 return (wxView *) NULL;
1364 }
1365 if (n == 1)
1366 {
1367 wxDocTemplate *temp = templates[0];
1368 delete[] templates;
1369 wxView *view = temp->CreateView(doc, flags);
1370 if (view)
1371 view->SetViewName(temp->GetViewName());
1372 return view;
1373 }
1374
1375 wxDocTemplate *temp = SelectViewType(templates, n);
1376 delete[] templates;
1377 if (temp)
1378 {
1379 wxView *view = temp->CreateView(doc, flags);
1380 if (view)
1381 view->SetViewName(temp->GetViewName());
1382 return view;
1383 }
1384 else
1385 return (wxView *) NULL;
1386}
1387
1388// Not yet implemented
1389void wxDocManager::DeleteTemplate(wxDocTemplate *WXUNUSED(temp), long WXUNUSED(flags))
1390{
1391}
1392
1393// Not yet implemented
1394bool wxDocManager::FlushDoc(wxDocument *WXUNUSED(doc))
1395{
1396 return false;
1397}
1398
1399wxDocument *wxDocManager::GetCurrentDocument() const
1400{
1401 wxView *view = GetCurrentView();
1402 if (view)
1403 return view->GetDocument();
1404 else
1405 return (wxDocument *) NULL;
1406}
1407
1408// Make a default document name
1409bool wxDocManager::MakeDefaultName(wxString& name)
1410{
1411 name.Printf(_("unnamed%d"), m_defaultDocumentNameCounter);
1412 m_defaultDocumentNameCounter++;
1413
1414 return true;
1415}
1416
1417// Make a frame title (override this to do something different)
1418// If docName is empty, a document is not currently active.
1419wxString wxDocManager::MakeFrameTitle(wxDocument* doc)
1420{
1421 wxString appName = wxTheApp->GetAppName();
1422 wxString title;
1423 if (!doc)
1424 title = appName;
1425 else
1426 {
1427 wxString docName;
1428 doc->GetPrintableName(docName);
1429 title = docName + wxString(_(" - ")) + appName;
1430 }
1431 return title;
1432}
1433
1434
1435// Not yet implemented
1436wxDocTemplate *wxDocManager::MatchTemplate(const wxString& WXUNUSED(path))
1437{
1438 return (wxDocTemplate *) NULL;
1439}
1440
1441// File history management
1442void wxDocManager::AddFileToHistory(const wxString& file)
1443{
1444 if (m_fileHistory)
1445 m_fileHistory->AddFileToHistory(file);
1446}
1447
1448void wxDocManager::RemoveFileFromHistory(size_t i)
1449{
1450 if (m_fileHistory)
1451 m_fileHistory->RemoveFileFromHistory(i);
1452}
1453
1454wxString wxDocManager::GetHistoryFile(size_t i) const
1455{
1456 wxString histFile;
1457
1458 if (m_fileHistory)
1459 histFile = m_fileHistory->GetHistoryFile(i);
1460
1461 return histFile;
1462}
1463
1464void wxDocManager::FileHistoryUseMenu(wxMenu *menu)
1465{
1466 if (m_fileHistory)
1467 m_fileHistory->UseMenu(menu);
1468}
1469
1470void wxDocManager::FileHistoryRemoveMenu(wxMenu *menu)
1471{
1472 if (m_fileHistory)
1473 m_fileHistory->RemoveMenu(menu);
1474}
1475
1476#if wxUSE_CONFIG
1477void wxDocManager::FileHistoryLoad(wxConfigBase& config)
1478{
1479 if (m_fileHistory)
1480 m_fileHistory->Load(config);
1481}
1482
1483void wxDocManager::FileHistorySave(wxConfigBase& config)
1484{
1485 if (m_fileHistory)
1486 m_fileHistory->Save(config);
1487}
1488#endif
1489
1490void wxDocManager::FileHistoryAddFilesToMenu(wxMenu* menu)
1491{
1492 if (m_fileHistory)
1493 m_fileHistory->AddFilesToMenu(menu);
1494}
1495
1496void wxDocManager::FileHistoryAddFilesToMenu()
1497{
1498 if (m_fileHistory)
1499 m_fileHistory->AddFilesToMenu();
1500}
1501
1502size_t wxDocManager::GetHistoryFilesCount() const
1503{
1504 return m_fileHistory ? m_fileHistory->GetCount() : 0;
1505}
1506
1507
1508// Find out the document template via matching in the document file format
1509// against that of the template
1510wxDocTemplate *wxDocManager::FindTemplateForPath(const wxString& path)
1511{
1512 wxDocTemplate *theTemplate = (wxDocTemplate *) NULL;
1513
1514 // Find the template which this extension corresponds to
1515 for (size_t i = 0; i < m_templates.GetCount(); i++)
1516 {
1517 wxDocTemplate *temp = (wxDocTemplate *)m_templates.Item(i)->GetData();
1518 if ( temp->FileMatchesTemplate(path) )
1519 {
1520 theTemplate = temp;
1521 break;
1522 }
1523 }
1524 return theTemplate;
1525}
1526
1527// Try to get a more suitable parent frame than the top window,
1528// for selection dialogs. Otherwise you may get an unexpected
1529// window being activated when a dialog is shown.
1530static wxWindow* wxFindSuitableParent()
1531{
1532 wxWindow* parent = wxTheApp->GetTopWindow();
1533
1534 wxWindow* focusWindow = wxWindow::FindFocus();
1535 if (focusWindow)
1536 {
1537 while (focusWindow &&
1538 !focusWindow->IsKindOf(CLASSINFO(wxDialog)) &&
1539 !focusWindow->IsKindOf(CLASSINFO(wxFrame)))
1540
1541 focusWindow = focusWindow->GetParent();
1542
1543 if (focusWindow)
1544 parent = focusWindow;
1545 }
1546 return parent;
1547}
1548
1549// Prompts user to open a file, using file specs in templates.
1550// Must extend the file selector dialog or implement own; OR
1551// match the extension to the template extension.
1552
1553wxDocTemplate *wxDocManager::SelectDocumentPath(wxDocTemplate **templates,
1554#if defined(__WXMSW__) || defined(__WXGTK__) || defined(__WXMAC__)
1555 int noTemplates,
1556#else
1557 int WXUNUSED(noTemplates),
1558#endif
1559 wxString& path,
1560 long WXUNUSED(flags),
1561 bool WXUNUSED(save))
1562{
1563 // We can only have multiple filters in Windows and GTK
1564#if defined(__WXMSW__) || defined(__WXGTK__) || defined(__WXMAC__)
1565 wxString descrBuf;
1566
1567 int i;
1568 for (i = 0; i < noTemplates; i++)
1569 {
1570 if (templates[i]->IsVisible())
1571 {
1572 // add a '|' to separate this filter from the previous one
1573 if ( !descrBuf.empty() )
1574 descrBuf << wxT('|');
1575
1576 descrBuf << templates[i]->GetDescription()
1577 << wxT(" (") << templates[i]->GetFileFilter() << wxT(") |")
1578 << templates[i]->GetFileFilter();
1579 }
1580 }
1581#else
1582 wxString descrBuf = wxT("*.*");
1583#endif
1584
1585 int FilterIndex = -1;
1586
1587 wxWindow* parent = wxFindSuitableParent();
1588
1589 wxString pathTmp = wxFileSelectorEx(_("Select a file"),
1590 m_lastDirectory,
1591 wxEmptyString,
1592 &FilterIndex,
1593 descrBuf,
1594 0,
1595 parent);
1596
1597 wxDocTemplate *theTemplate = (wxDocTemplate *)NULL;
1598 if (!pathTmp.empty())
1599 {
1600 if (!wxFileExists(pathTmp))
1601 {
1602 wxString msgTitle;
1603 if (!wxTheApp->GetAppName().empty())
1604 msgTitle = wxTheApp->GetAppName();
1605 else
1606 msgTitle = wxString(_("File error"));
1607
1608 (void)wxMessageBox(_("Sorry, could not open this file."), msgTitle, wxOK | wxICON_EXCLAMATION,
1609 parent);
1610
1611 path = wxEmptyString;
1612 return (wxDocTemplate *) NULL;
1613 }
1614 m_lastDirectory = wxPathOnly(pathTmp);
1615
1616 path = pathTmp;
1617
1618 // first choose the template using the extension, if this fails (i.e.
1619 // wxFileSelectorEx() didn't fill it), then use the path
1620 if ( FilterIndex != -1 )
1621 theTemplate = templates[FilterIndex];
1622 if ( !theTemplate )
1623 theTemplate = FindTemplateForPath(path);
1624 if ( !theTemplate )
1625 {
1626 // Since we do not add files with non-default extensions to the FileHistory this
1627 // can only happen if the application changes the allowed templates in runtime.
1628 (void)wxMessageBox(_("Sorry, the format for this file is unknown."),
1629 _("Open File"),
1630 wxOK | wxICON_EXCLAMATION, wxFindSuitableParent());
1631 }
1632 }
1633 else
1634 {
1635 path = wxEmptyString;
1636 }
1637
1638 return theTemplate;
1639}
1640
1641wxDocTemplate *wxDocManager::SelectDocumentType(wxDocTemplate **templates,
1642 int noTemplates, bool sort)
1643{
1644 wxArrayString strings;
1645 wxDocTemplate **data = new wxDocTemplate *[noTemplates];
1646 int i;
1647 int n = 0;
1648
1649 for (i = 0; i < noTemplates; i++)
1650 {
1651 if (templates[i]->IsVisible())
1652 {
1653 int j;
1654 bool want = true;
1655 for (j = 0; j < n; j++)
1656 {
1657 //filter out NOT unique documents + view combinations
1658 if ( templates[i]->m_docTypeName == data[j]->m_docTypeName &&
1659 templates[i]->m_viewTypeName == data[j]->m_viewTypeName
1660 )
1661 want = false;
1662 }
1663
1664 if ( want )
1665 {
1666 strings.Add(templates[i]->m_description);
1667
1668 data[n] = templates[i];
1669 n ++;
1670 }
1671 }
1672 } // for
1673
1674 if (sort)
1675 {
1676 strings.Sort(); // ascending sort
1677 // Yes, this will be slow, but template lists
1678 // are typically short.
1679 int j;
1680 n = strings.Count();
1681 for (i = 0; i < n; i++)
1682 {
1683 for (j = 0; j < noTemplates; j++)
1684 {
1685 if (strings[i] == templates[j]->m_description)
1686 data[i] = templates[j];
1687 }
1688 }
1689 }
1690
1691 wxDocTemplate *theTemplate;
1692
1693 switch ( n )
1694 {
1695 case 0:
1696 // no visible templates, hence nothing to choose from
1697 theTemplate = NULL;
1698 break;
1699
1700 case 1:
1701 // don't propose the user to choose if he heas no choice
1702 theTemplate = data[0];
1703 break;
1704
1705 default:
1706 // propose the user to choose one of several
1707 theTemplate = (wxDocTemplate *)wxGetSingleChoiceData
1708 (
1709 _("Select a document template"),
1710 _("Templates"),
1711 strings,
1712 (void **)data,
1713 wxFindSuitableParent()
1714 );
1715 }
1716
1717 delete[] data;
1718
1719 return theTemplate;
1720}
1721
1722wxDocTemplate *wxDocManager::SelectViewType(wxDocTemplate **templates,
1723 int noTemplates, bool sort)
1724{
1725 wxArrayString strings;
1726 wxDocTemplate **data = new wxDocTemplate *[noTemplates];
1727 int i;
1728 int n = 0;
1729
1730 for (i = 0; i < noTemplates; i++)
1731 {
1732 wxDocTemplate *templ = templates[i];
1733 if ( templ->IsVisible() && !templ->GetViewName().empty() )
1734 {
1735 int j;
1736 bool want = true;
1737 for (j = 0; j < n; j++)
1738 {
1739 //filter out NOT unique views
1740 if ( templates[i]->m_viewTypeName == data[j]->m_viewTypeName )
1741 want = false;
1742 }
1743
1744 if ( want )
1745 {
1746 strings.Add(templ->m_viewTypeName);
1747 data[n] = templ;
1748 n ++;
1749 }
1750 }
1751 }
1752
1753 if (sort)
1754 {
1755 strings.Sort(); // ascending sort
1756 // Yes, this will be slow, but template lists
1757 // are typically short.
1758 int j;
1759 n = strings.Count();
1760 for (i = 0; i < n; i++)
1761 {
1762 for (j = 0; j < noTemplates; j++)
1763 {
1764 if (strings[i] == templates[j]->m_viewTypeName)
1765 data[i] = templates[j];
1766 }
1767 }
1768 }
1769
1770 wxDocTemplate *theTemplate;
1771
1772 // the same logic as above
1773 switch ( n )
1774 {
1775 case 0:
1776 theTemplate = (wxDocTemplate *)NULL;
1777 break;
1778
1779 case 1:
1780 theTemplate = data[0];
1781 break;
1782
1783 default:
1784 theTemplate = (wxDocTemplate *)wxGetSingleChoiceData
1785 (
1786 _("Select a document view"),
1787 _("Views"),
1788 strings,
1789 (void **)data,
1790 wxFindSuitableParent()
1791 );
1792
1793 }
1794
1795 delete[] data;
1796 return theTemplate;
1797}
1798
1799void wxDocManager::AssociateTemplate(wxDocTemplate *temp)
1800{
1801 if (!m_templates.Member(temp))
1802 m_templates.Append(temp);
1803}
1804
1805void wxDocManager::DisassociateTemplate(wxDocTemplate *temp)
1806{
1807 m_templates.DeleteObject(temp);
1808}
1809
1810// Add and remove a document from the manager's list
1811void wxDocManager::AddDocument(wxDocument *doc)
1812{
1813 if (!m_docs.Member(doc))
1814 m_docs.Append(doc);
1815}
1816
1817void wxDocManager::RemoveDocument(wxDocument *doc)
1818{
1819 m_docs.DeleteObject(doc);
1820}
1821
1822// Views or windows should inform the document manager
1823// when a view is going in or out of focus
1824void wxDocManager::ActivateView(wxView *view, bool activate)
1825{
1826 if ( activate )
1827 {
1828 m_currentView = view;
1829 }
1830 else // deactivate
1831 {
1832 if ( m_currentView == view )
1833 {
1834 // don't keep stale pointer
1835 m_currentView = (wxView *) NULL;
1836 }
1837 }
1838}
1839
1840// ----------------------------------------------------------------------------
1841// Default document child frame
1842// ----------------------------------------------------------------------------
1843
1844BEGIN_EVENT_TABLE(wxDocChildFrame, wxFrame)
1845 EVT_ACTIVATE(wxDocChildFrame::OnActivate)
1846 EVT_CLOSE(wxDocChildFrame::OnCloseWindow)
1847END_EVENT_TABLE()
1848
1849wxDocChildFrame::wxDocChildFrame(wxDocument *doc,
1850 wxView *view,
1851 wxFrame *frame,
1852 wxWindowID id,
1853 const wxString& title,
1854 const wxPoint& pos,
1855 const wxSize& size,
1856 long style,
1857 const wxString& name)
1858 : wxFrame(frame, id, title, pos, size, style, name)
1859{
1860 m_childDocument = doc;
1861 m_childView = view;
1862 if (view)
1863 view->SetFrame(this);
1864}
1865
1866// Extend event processing to search the view's event table
1867bool wxDocChildFrame::ProcessEvent(wxEvent& event)
1868{
1869 if (m_childView)
1870 m_childView->Activate(true);
1871
1872 if ( !m_childView || ! m_childView->ProcessEvent(event) )
1873 {
1874 // Only hand up to the parent if it's a menu command
1875 if (!event.IsKindOf(CLASSINFO(wxCommandEvent)) || !GetParent() || !GetParent()->ProcessEvent(event))
1876 return wxEvtHandler::ProcessEvent(event);
1877 else
1878 return true;
1879 }
1880 else
1881 return true;
1882}
1883
1884void wxDocChildFrame::OnActivate(wxActivateEvent& event)
1885{
1886 wxFrame::OnActivate(event);
1887
1888 if (m_childView)
1889 m_childView->Activate(event.GetActive());
1890}
1891
1892void wxDocChildFrame::OnCloseWindow(wxCloseEvent& event)
1893{
1894 if (m_childView)
1895 {
1896 bool ans = event.CanVeto()
1897 ? m_childView->Close(false) // false means don't delete associated window
1898 : true; // Must delete.
1899
1900 if (ans)
1901 {
1902 m_childView->Activate(false);
1903 delete m_childView;
1904 m_childView = (wxView *) NULL;
1905 m_childDocument = (wxDocument *) NULL;
1906
1907 this->Destroy();
1908 }
1909 else
1910 event.Veto();
1911 }
1912 else
1913 event.Veto();
1914}
1915
1916// ----------------------------------------------------------------------------
1917// Default parent frame
1918// ----------------------------------------------------------------------------
1919
1920BEGIN_EVENT_TABLE(wxDocParentFrame, wxFrame)
1921 EVT_MENU(wxID_EXIT, wxDocParentFrame::OnExit)
1922 EVT_MENU_RANGE(wxID_FILE1, wxID_FILE9, wxDocParentFrame::OnMRUFile)
1923 EVT_CLOSE(wxDocParentFrame::OnCloseWindow)
1924END_EVENT_TABLE()
1925
1926wxDocParentFrame::wxDocParentFrame(wxDocManager *manager,
1927 wxFrame *frame,
1928 wxWindowID id,
1929 const wxString& title,
1930 const wxPoint& pos,
1931 const wxSize& size,
1932 long style,
1933 const wxString& name)
1934 : wxFrame(frame, id, title, pos, size, style, name)
1935{
1936 m_docManager = manager;
1937}
1938
1939void wxDocParentFrame::OnExit(wxCommandEvent& WXUNUSED(event))
1940{
1941 Close();
1942}
1943
1944void wxDocParentFrame::OnMRUFile(wxCommandEvent& event)
1945{
1946 int n = event.GetId() - wxID_FILE1; // the index in MRU list
1947 wxString filename(m_docManager->GetHistoryFile(n));
1948 if ( !filename.empty() )
1949 {
1950 // verify that the file exists before doing anything else
1951 if ( wxFile::Exists(filename) )
1952 {
1953 // try to open it
1954 if (!m_docManager->CreateDocument(filename, wxDOC_SILENT))
1955 {
1956 // remove the file from the MRU list. The user should already be notified.
1957 m_docManager->RemoveFileFromHistory(n);
1958
1959 wxLogError(_("The file '%s' couldn't be opened.\nIt has been removed from the most recently used files list."),
1960 filename.c_str());
1961 }
1962 }
1963 else
1964 {
1965 // remove the bogus filename from the MRU list and notify the user
1966 // about it
1967 m_docManager->RemoveFileFromHistory(n);
1968
1969 wxLogError(_("The file '%s' doesn't exist and couldn't be opened.\nIt has been removed from the most recently used files list."),
1970 filename.c_str());
1971 }
1972 }
1973}
1974
1975// Extend event processing to search the view's event table
1976bool wxDocParentFrame::ProcessEvent(wxEvent& event)
1977{
1978 // Try the document manager, then do default processing
1979 if (!m_docManager || !m_docManager->ProcessEvent(event))
1980 return wxEvtHandler::ProcessEvent(event);
1981 else
1982 return true;
1983}
1984
1985// Define the behaviour for the frame closing
1986// - must delete all frames except for the main one.
1987void wxDocParentFrame::OnCloseWindow(wxCloseEvent& event)
1988{
1989 if (m_docManager->Clear(!event.CanVeto()))
1990 {
1991 this->Destroy();
1992 }
1993 else
1994 event.Veto();
1995}
1996
1997#if wxUSE_PRINTING_ARCHITECTURE
1998
1999wxDocPrintout::wxDocPrintout(wxView *view, const wxString& title)
2000 : wxPrintout(title)
2001{
2002 m_printoutView = view;
2003}
2004
2005bool wxDocPrintout::OnPrintPage(int WXUNUSED(page))
2006{
2007 wxDC *dc = GetDC();
2008
2009 // Get the logical pixels per inch of screen and printer
2010 int ppiScreenX, ppiScreenY;
2011 GetPPIScreen(&ppiScreenX, &ppiScreenY);
2012 wxUnusedVar(ppiScreenY);
2013 int ppiPrinterX, ppiPrinterY;
2014 GetPPIPrinter(&ppiPrinterX, &ppiPrinterY);
2015 wxUnusedVar(ppiPrinterY);
2016
2017 // This scales the DC so that the printout roughly represents the
2018 // the screen scaling. The text point size _should_ be the right size
2019 // but in fact is too small for some reason. This is a detail that will
2020 // need to be addressed at some point but can be fudged for the
2021 // moment.
2022 float scale = (float)((float)ppiPrinterX/(float)ppiScreenX);
2023
2024 // Now we have to check in case our real page size is reduced
2025 // (e.g. because we're drawing to a print preview memory DC)
2026 int pageWidth, pageHeight;
2027 int w, h;
2028 dc->GetSize(&w, &h);
2029 GetPageSizePixels(&pageWidth, &pageHeight);
2030 wxUnusedVar(pageHeight);
2031
2032 // If printer pageWidth == current DC width, then this doesn't
2033 // change. But w might be the preview bitmap width, so scale down.
2034 float overallScale = scale * (float)(w/(float)pageWidth);
2035 dc->SetUserScale(overallScale, overallScale);
2036
2037 if (m_printoutView)
2038 {
2039 m_printoutView->OnDraw(dc);
2040 }
2041 return true;
2042}
2043
2044bool wxDocPrintout::HasPage(int pageNum)
2045{
2046 return (pageNum == 1);
2047}
2048
2049bool wxDocPrintout::OnBeginDocument(int startPage, int endPage)
2050{
2051 if (!wxPrintout::OnBeginDocument(startPage, endPage))
2052 return false;
2053
2054 return true;
2055}
2056
2057void wxDocPrintout::GetPageInfo(int *minPage, int *maxPage, int *selPageFrom, int *selPageTo)
2058{
2059 *minPage = 1;
2060 *maxPage = 1;
2061 *selPageFrom = 1;
2062 *selPageTo = 1;
2063}
2064
2065#endif // wxUSE_PRINTING_ARCHITECTURE
2066
2067// ----------------------------------------------------------------------------
2068// File history processor
2069// ----------------------------------------------------------------------------
2070
2071static inline wxChar* MYcopystring(const wxString& s)
2072{
2073 wxChar* copy = new wxChar[s.length() + 1];
2074 return wxStrcpy(copy, s.c_str());
2075}
2076
2077static inline wxChar* MYcopystring(const wxChar* s)
2078{
2079 wxChar* copy = new wxChar[wxStrlen(s) + 1];
2080 return wxStrcpy(copy, s);
2081}
2082
2083wxFileHistory::wxFileHistory(size_t maxFiles, wxWindowID idBase)
2084{
2085 m_fileMaxFiles = maxFiles;
2086 m_idBase = idBase;
2087 m_fileHistoryN = 0;
2088 m_fileHistory = new wxChar *[m_fileMaxFiles];
2089}
2090
2091wxFileHistory::~wxFileHistory()
2092{
2093 size_t i;
2094 for (i = 0; i < m_fileHistoryN; i++)
2095 delete[] m_fileHistory[i];
2096 delete[] m_fileHistory;
2097}
2098
2099// File history management
2100void wxFileHistory::AddFileToHistory(const wxString& file)
2101{
2102 size_t i;
2103
2104 // Check we don't already have this file
2105 for (i = 0; i < m_fileHistoryN; i++)
2106 {
2107#if defined( __WXMSW__ ) // Add any other OSes with case insensitive file names
2108 wxString testString;
2109 if ( m_fileHistory[i] )
2110 testString = m_fileHistory[i];
2111 if ( m_fileHistory[i] && ( file.Lower() == testString.Lower() ) )
2112#else
2113 if ( m_fileHistory[i] && ( file == m_fileHistory[i] ) )
2114#endif
2115 {
2116 // we do have it, move it to the top of the history
2117 RemoveFileFromHistory (i);
2118 AddFileToHistory (file);
2119 return;
2120 }
2121 }
2122
2123 // if we already have a full history, delete the one at the end
2124 if ( m_fileMaxFiles == m_fileHistoryN )
2125 {
2126 RemoveFileFromHistory (m_fileHistoryN - 1);
2127 AddFileToHistory (file);
2128 return;
2129 }
2130
2131 // Add to the project file history:
2132 // Move existing files (if any) down so we can insert file at beginning.
2133 if (m_fileHistoryN < m_fileMaxFiles)
2134 {
2135 wxList::compatibility_iterator node = m_fileMenus.GetFirst();
2136 while (node)
2137 {
2138 wxMenu* menu = (wxMenu*) node->GetData();
2139 if ( m_fileHistoryN == 0 && menu->GetMenuItemCount() )
2140 {
2141 menu->AppendSeparator();
2142 }
2143 menu->Append(m_idBase+m_fileHistoryN, _("[EMPTY]"));
2144 node = node->GetNext();
2145 }
2146 m_fileHistoryN ++;
2147 }
2148 // Shuffle filenames down
2149 for (i = (m_fileHistoryN-1); i > 0; i--)
2150 {
2151 m_fileHistory[i] = m_fileHistory[i-1];
2152 }
2153 m_fileHistory[0] = MYcopystring(file);
2154
2155 // this is the directory of the last opened file
2156 wxString pathCurrent;
2157 wxSplitPath( m_fileHistory[0], &pathCurrent, NULL, NULL );
2158 for (i = 0; i < m_fileHistoryN; i++)
2159 {
2160 if ( m_fileHistory[i] )
2161 {
2162 // if in same directory just show the filename; otherwise the full
2163 // path
2164 wxString pathInMenu, path, filename, ext;
2165 wxSplitPath( m_fileHistory[i], &path, &filename, &ext );
2166 if ( path == pathCurrent )
2167 {
2168 pathInMenu = filename;
2169 if ( !ext.empty() )
2170 pathInMenu = pathInMenu + wxFILE_SEP_EXT + ext;
2171 }
2172 else
2173 {
2174 // absolute path; could also set relative path
2175 pathInMenu = m_fileHistory[i];
2176 }
2177
2178 // we need to quote '&' characters which are used for mnemonics
2179 pathInMenu.Replace(_T("&"), _T("&&"));
2180 wxString buf;
2181 buf.Printf(s_MRUEntryFormat, i + 1, pathInMenu.c_str());
2182 wxList::compatibility_iterator node = m_fileMenus.GetFirst();
2183 while (node)
2184 {
2185 wxMenu* menu = (wxMenu*) node->GetData();
2186 menu->SetLabel(m_idBase + i, buf);
2187 node = node->GetNext();
2188 }
2189 }
2190 }
2191}
2192
2193void wxFileHistory::RemoveFileFromHistory(size_t i)
2194{
2195 wxCHECK_RET( i < m_fileHistoryN,
2196 wxT("invalid index in wxFileHistory::RemoveFileFromHistory") );
2197
2198 // delete the element from the array (could use memmove() too...)
2199 delete [] m_fileHistory[i];
2200
2201 size_t j;
2202 for ( j = i; j < m_fileHistoryN - 1; j++ )
2203 {
2204 m_fileHistory[j] = m_fileHistory[j + 1];
2205 }
2206
2207 wxList::compatibility_iterator node = m_fileMenus.GetFirst();
2208 while ( node )
2209 {
2210 wxMenu* menu = (wxMenu*) node->GetData();
2211
2212 // shuffle filenames up
2213 wxString buf;
2214 for ( j = i; j < m_fileHistoryN - 1; j++ )
2215 {
2216 buf.Printf(s_MRUEntryFormat, j + 1, m_fileHistory[j]);
2217 menu->SetLabel(m_idBase + j, buf);
2218 }
2219
2220 node = node->GetNext();
2221
2222 // delete the last menu item which is unused now
2223 wxWindowID lastItemId = m_idBase + wx_truncate_cast(wxWindowID, m_fileHistoryN) - 1;
2224 if (menu->FindItem(lastItemId))
2225 {
2226 menu->Delete(lastItemId);
2227 }
2228
2229 // delete the last separator too if no more files are left
2230 if ( m_fileHistoryN == 1 )
2231 {
2232 wxMenuItemList::compatibility_iterator nodeLast = menu->GetMenuItems().GetLast();
2233 if ( nodeLast )
2234 {
2235 wxMenuItem *menuItem = nodeLast->GetData();
2236 if ( menuItem->IsSeparator() )
2237 {
2238 menu->Delete(menuItem);
2239 }
2240 //else: should we search backwards for the last separator?
2241 }
2242 //else: menu is empty somehow
2243 }
2244 }
2245
2246 m_fileHistoryN--;
2247}
2248
2249wxString wxFileHistory::GetHistoryFile(size_t i) const
2250{
2251 wxString s;
2252 if ( i < m_fileHistoryN )
2253 {
2254 s = m_fileHistory[i];
2255 }
2256 else
2257 {
2258 wxFAIL_MSG( wxT("bad index in wxFileHistory::GetHistoryFile") );
2259 }
2260
2261 return s;
2262}
2263
2264void wxFileHistory::UseMenu(wxMenu *menu)
2265{
2266 if (!m_fileMenus.Member(menu))
2267 m_fileMenus.Append(menu);
2268}
2269
2270void wxFileHistory::RemoveMenu(wxMenu *menu)
2271{
2272 m_fileMenus.DeleteObject(menu);
2273}
2274
2275#if wxUSE_CONFIG
2276void wxFileHistory::Load(wxConfigBase& config)
2277{
2278 m_fileHistoryN = 0;
2279 wxString buf;
2280 buf.Printf(wxT("file%d"), (int)m_fileHistoryN+1);
2281 wxString historyFile;
2282 while ((m_fileHistoryN < m_fileMaxFiles) && config.Read(buf, &historyFile) && (!historyFile.empty()))
2283 {
2284 m_fileHistory[m_fileHistoryN] = MYcopystring((const wxChar*) historyFile);
2285 m_fileHistoryN ++;
2286 buf.Printf(wxT("file%d"), (int)m_fileHistoryN+1);
2287 historyFile = wxEmptyString;
2288 }
2289 AddFilesToMenu();
2290}
2291
2292void wxFileHistory::Save(wxConfigBase& config)
2293{
2294 size_t i;
2295 for (i = 0; i < m_fileMaxFiles; i++)
2296 {
2297 wxString buf;
2298 buf.Printf(wxT("file%d"), (int)i+1);
2299 if (i < m_fileHistoryN)
2300 config.Write(buf, wxString(m_fileHistory[i]));
2301 else
2302 config.Write(buf, wxEmptyString);
2303 }
2304}
2305#endif // wxUSE_CONFIG
2306
2307void wxFileHistory::AddFilesToMenu()
2308{
2309 if (m_fileHistoryN > 0)
2310 {
2311 wxList::compatibility_iterator node = m_fileMenus.GetFirst();
2312 while (node)
2313 {
2314 wxMenu* menu = (wxMenu*) node->GetData();
2315 if (menu->GetMenuItemCount())
2316 {
2317 menu->AppendSeparator();
2318 }
2319
2320 size_t i;
2321 for (i = 0; i < m_fileHistoryN; i++)
2322 {
2323 if (m_fileHistory[i])
2324 {
2325 wxString buf;
2326 buf.Printf(s_MRUEntryFormat, i+1, m_fileHistory[i]);
2327 menu->Append(m_idBase+i, buf);
2328 }
2329 }
2330 node = node->GetNext();
2331 }
2332 }
2333}
2334
2335void wxFileHistory::AddFilesToMenu(wxMenu* menu)
2336{
2337 if (m_fileHistoryN > 0)
2338 {
2339 if (menu->GetMenuItemCount())
2340 {
2341 menu->AppendSeparator();
2342 }
2343
2344 size_t i;
2345 for (i = 0; i < m_fileHistoryN; i++)
2346 {
2347 if (m_fileHistory[i])
2348 {
2349 wxString buf;
2350 buf.Printf(s_MRUEntryFormat, i+1, m_fileHistory[i]);
2351 menu->Append(m_idBase+i, buf);
2352 }
2353 }
2354 }
2355}
2356
2357// ----------------------------------------------------------------------------
2358// Permits compatibility with existing file formats and functions that
2359// manipulate files directly
2360// ----------------------------------------------------------------------------
2361
2362#if wxUSE_STD_IOSTREAM
2363
2364bool wxTransferFileToStream(const wxString& filename, wxSTD ostream& stream)
2365{
2366 wxFFile file(filename, _T("rb"));
2367 if ( !file.IsOpened() )
2368 return false;
2369
2370 char buf[4096];
2371
2372 size_t nRead;
2373 do
2374 {
2375 nRead = file.Read(buf, WXSIZEOF(buf));
2376 if ( file.Error() )
2377 return false;
2378
2379 stream.write(buf, nRead);
2380 if ( !stream )
2381 return false;
2382 }
2383 while ( !file.Eof() );
2384
2385 return true;
2386}
2387
2388bool wxTransferStreamToFile(wxSTD istream& stream, const wxString& filename)
2389{
2390 wxFFile file(filename, _T("wb"));
2391 if ( !file.IsOpened() )
2392 return false;
2393
2394 char buf[4096];
2395 do
2396 {
2397 stream.read(buf, WXSIZEOF(buf));
2398 if ( !stream.bad() ) // fail may be set on EOF, don't use operator!()
2399 {
2400 if ( !file.Write(buf, stream.gcount()) )
2401 return false;
2402 }
2403 }
2404 while ( !stream.eof() );
2405
2406 return true;
2407}
2408
2409#else // !wxUSE_STD_IOSTREAM
2410
2411bool wxTransferFileToStream(const wxString& filename, wxOutputStream& stream)
2412{
2413 wxFFile file(filename, _T("rb"));
2414 if ( !file.IsOpened() )
2415 return false;
2416
2417 char buf[4096];
2418
2419 size_t nRead;
2420 do
2421 {
2422 nRead = file.Read(buf, WXSIZEOF(buf));
2423 if ( file.Error() )
2424 return false;
2425
2426 stream.Write(buf, nRead);
2427 if ( !stream )
2428 return false;
2429 }
2430 while ( !file.Eof() );
2431
2432 return true;
2433}
2434
2435bool wxTransferStreamToFile(wxInputStream& stream, const wxString& filename)
2436{
2437 wxFFile file(filename, _T("wb"));
2438 if ( !file.IsOpened() )
2439 return false;
2440
2441 char buf[4096];
2442 do
2443 {
2444 stream.Read(buf, WXSIZEOF(buf));
2445
2446 const size_t nRead = stream.LastRead();
2447 if ( !nRead || !file.Write(buf, nRead) )
2448 return false;
2449 }
2450 while ( !stream.Eof() );
2451
2452 return true;
2453}
2454
2455#endif // wxUSE_STD_IOSTREAM/!wxUSE_STD_IOSTREAM
2456
2457#endif // wxUSE_DOC_VIEW_ARCHITECTURE