OSDN Git Service

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