X-Git-Url: https://git.saurik.com/wxWidgets.git/blobdiff_plain/56d7679d3ef0bf4d761d791df00ce17ecd662014..1d8cdcf2e39810b8e6128ac8a504cc206068f58c:/samples/docview/doc.cpp diff --git a/samples/docview/doc.cpp b/samples/docview/doc.cpp index fb915e6465..cf53892ea2 100644 --- a/samples/docview/doc.cpp +++ b/samples/docview/doc.cpp @@ -1,363 +1,305 @@ ///////////////////////////////////////////////////////////////////////////// -// Name: doc.cpp +// Name: samples/docview/doc.cpp // Purpose: Implements document functionality // Author: Julian Smart -// Modified by: +// Modified by: Vadim Zeitlin: merge with the MDI version and general cleanup // Created: 04/01/98 // RCS-ID: $Id$ -// Copyright: (c) Julian Smart and Markus Holzem -// Licence: wxWindows license +// Copyright: (c) 1998 Julian Smart +// (c) 2008 Vadim Zeitlin +// Licence: wxWindows license ///////////////////////////////////////////////////////////////////////////// -#ifdef __GNUG__ -// #pragma implementation -#endif +// ---------------------------------------------------------------------------- +// headers +// ---------------------------------------------------------------------------- // For compilers that support precompilation, includes "wx/wx.h". #include "wx/wxprec.h" #ifdef __BORLANDC__ -#pragma hdrstop + #pragma hdrstop +#endif + +#if !wxUSE_DOC_VIEW_ARCHITECTURE + #error You must set wxUSE_DOC_VIEW_ARCHITECTURE to 1 in setup.h! #endif #ifndef WX_PRECOMP -#include "wx/wx.h" + #include "wx/wx.h" #endif -#include "wx/txtstrm.h" -#if !wxUSE_DOC_VIEW_ARCHITECTURE -#error You must set wxUSE_DOC_VIEW_ARCHITECTURE to 1 in setup.h! +#if wxUSE_STD_IOSTREAM + #include "wx/ioswrap.h" +#else + #include "wx/txtstrm.h" #endif +#include "wx/wfstream.h" #include "doc.h" #include "view.h" +// ---------------------------------------------------------------------------- +// DrawingDocument implementation +// ---------------------------------------------------------------------------- + IMPLEMENT_DYNAMIC_CLASS(DrawingDocument, wxDocument) -DrawingDocument::DrawingDocument(void) +DocumentOstream& DrawingDocument::SaveObject(DocumentOstream& ostream) { +#if wxUSE_STD_IOSTREAM + DocumentOstream& stream = ostream; +#else + wxTextOutputStream stream(ostream); +#endif + + wxDocument::SaveObject(ostream); + + const wxInt32 count = m_doodleSegments.size(); + stream << count << '\n'; + + for ( int n = 0; n < count; n++ ) + { + m_doodleSegments[n].SaveObject(ostream); + stream << '\n'; + } + + return ostream; } -DrawingDocument::~DrawingDocument(void) +DocumentIstream& DrawingDocument::LoadObject(DocumentIstream& istream) { - doodleSegments.DeleteContents(TRUE); +#if wxUSE_STD_IOSTREAM + DocumentIstream& stream = istream; +#else + wxTextInputStream stream(istream); +#endif + + wxDocument::LoadObject(istream); + + wxInt32 count = 0; + stream >> count; + + for ( int n = 0; n < count; n++ ) + { + DoodleSegment segment; + segment.LoadObject(istream); + m_doodleSegments.push_back(segment); + } + + return istream; } -#if wxUSE_STD_IOSTREAM -ostream& DrawingDocument::SaveObject(ostream& stream) +void DrawingDocument::DoUpdate() { - wxDocument::SaveObject(stream); - - stream << doodleSegments.Number() << '\n'; - wxNode *node = doodleSegments.First(); - while (node) - { - DoodleSegment *segment = (DoodleSegment *)node->Data(); - segment->SaveObject(stream); - stream << '\n'; - - node = node->Next(); - } - return stream; + Modify(true); + UpdateAllViews(); } -#else -bool DrawingDocument::SaveObject(wxOutputStream &stream) + +void DrawingDocument::AddDoodleSegment(const DoodleSegment& segment) { - wxDocument::SaveObject(stream); - - wxTextOutputStream text_stream( stream ); - - wxInt32 n = doodleSegments.Number(); - text_stream << n << "\n"; - - wxNode *node = doodleSegments.First(); - while (node) - { - DoodleSegment *segment = (DoodleSegment *)node->Data(); - segment->SaveObject(stream); - text_stream << "\n"; - - node = node->Next(); - } - return TRUE; + m_doodleSegments.push_back(segment); + + DoUpdate(); } -#endif -#if wxUSE_STD_IOSTREAM -istream& DrawingDocument::LoadObject(istream& stream) +bool DrawingDocument::PopLastSegment(DoodleSegment *segment) { - wxDocument::LoadObject(stream); + if ( m_doodleSegments.empty() ) + return false; + + if ( segment ) + *segment = m_doodleSegments.back(); - int n = 0; - stream >> n; + m_doodleSegments.pop_back(); - for (int i = 0; i < n; i++) - { - DoodleSegment *segment = new DoodleSegment; - segment->LoadObject(stream); - doodleSegments.Append(segment); - } + DoUpdate(); - return stream; + return true; } -#else -bool DrawingDocument::LoadObject(wxInputStream& stream) + +// ---------------------------------------------------------------------------- +// DoodleSegment implementation +// ---------------------------------------------------------------------------- + +DocumentOstream& DoodleSegment::SaveObject(DocumentOstream& ostream) { - wxDocument::LoadObject(stream); +#if wxUSE_STD_IOSTREAM + DocumentOstream& stream = ostream; +#else + wxTextOutputStream stream(ostream); +#endif - wxTextInputStream text_stream( stream ); - - wxInt32 n = 0; - text_stream >> n; + const wxInt32 count = m_lines.size(); + stream << count << '\n'; - for (int i = 0; i < n; i++) - { - DoodleSegment *segment = new DoodleSegment; - segment->LoadObject(stream); - doodleSegments.Append(segment); - } + for ( int n = 0; n < count; n++ ) + { + const DoodleLine& line = m_lines[n]; + stream + << line.x1 << ' ' + << line.y1 << ' ' + << line.x2 << ' ' + << line.y2 << '\n'; + } - return TRUE; + return ostream; } -#endif -DoodleSegment::DoodleSegment(void) +DocumentIstream& DoodleSegment::LoadObject(DocumentIstream& istream) { +#if wxUSE_STD_IOSTREAM + DocumentIstream& stream = istream; +#else + wxTextInputStream stream(istream); +#endif + + wxInt32 count = 0; + stream >> count; + + for ( int n = 0; n < count; n++ ) + { + DoodleLine line; + stream + >> line.x1 + >> line.y1 + >> line.x2 + >> line.y2; + m_lines.push_back(line); + } + + return istream; } -DoodleSegment::DoodleSegment(DoodleSegment& seg) +// ---------------------------------------------------------------------------- +// wxTextDocument: wxDocument and wxTextCtrl married +// ---------------------------------------------------------------------------- + +IMPLEMENT_CLASS(wxTextDocument, wxDocument) + +bool wxTextDocument::OnCreate(const wxString& path, long flags) { - wxNode *node = seg.lines.First(); - while (node) - { - DoodleLine *line = (DoodleLine *)node->Data(); - DoodleLine *newLine = new DoodleLine; - newLine->x1 = line->x1; - newLine->y1 = line->y1; - newLine->x2 = line->x2; - newLine->y2 = line->y2; - - lines.Append(newLine); - - node = node->Next(); - } + if ( !wxDocument::OnCreate(path, flags) ) + return false; + + // subscribe to changes in the text control to update the document state + // when it's modified + GetTextCtrl()->Connect + ( + wxEVT_COMMAND_TEXT_UPDATED, + wxCommandEventHandler(wxTextDocument::OnTextChange), + NULL, + this + ); + + return true; } -DoodleSegment::~DoodleSegment(void) +// Since text windows have their own method for saving to/loading from files, +// we override DoSave/OpenDocument instead of Save/LoadObject +bool wxTextDocument::DoSaveDocument(const wxString& filename) { - lines.DeleteContents(TRUE); + return GetTextCtrl()->SaveFile(filename); } -#if wxUSE_STD_IOSTREAM -ostream& DoodleSegment::SaveObject(ostream& stream) +bool wxTextDocument::DoOpenDocument(const wxString& filename) { - stream << lines.Number() << '\n'; - wxNode *node = lines.First(); - while (node) - { - DoodleLine *line = (DoodleLine *)node->Data(); - stream << line->x1 << " " << line->y1 << " " << line->x2 << " " << line->y2 << "\n"; - node = node->Next(); - } - return stream; + if ( !GetTextCtrl()->LoadFile(filename) ) + return false; + + // we're not modified by the user yet + Modify(false); + + return true; } -#else -bool DoodleSegment::SaveObject(wxOutputStream& stream) + +bool wxTextDocument::IsModified() const { - wxTextOutputStream text_stream( stream ); - - wxInt32 n = lines.Number(); - text_stream << n << "\n"; - - wxNode *node = lines.First(); - while (node) - { - DoodleLine *line = (DoodleLine *)node->Data(); - text_stream << line->x1 << " " << - line->y1 << " " << - line->x2 << " " << - line->y2 << "\n"; - node = node->Next(); - } - return TRUE; + wxTextCtrl* wnd = GetTextCtrl(); + return wxDocument::IsModified() || (wnd && wnd->IsModified()); } -#endif -#if wxUSE_STD_IOSTREAM -istream& DoodleSegment::LoadObject(istream& stream) +void wxTextDocument::Modify(bool modified) { - int n = 0; - stream >> n; - - for (int i = 0; i < n; i++) - { - DoodleLine *line = new DoodleLine; - stream >> line->x1 >> line->y1 >> line->x2 >> line->y2; - lines.Append(line); - } - return stream; + wxDocument::Modify(modified); + + wxTextCtrl* wnd = GetTextCtrl(); + if (wnd && !modified) + { + wnd->DiscardEdits(); + } } -#else -bool DoodleSegment::LoadObject(wxInputStream& stream) + +void wxTextDocument::OnTextChange(wxCommandEvent& event) { - wxTextInputStream text_stream( stream ); - - wxInt32 n = 0; - text_stream >> n; - - for (int i = 0; i < n; i++) - { - DoodleLine *line = new DoodleLine; - text_stream >> line->x1 >> - line->y1 >> - line->x2 >> - line->y2; - lines.Append(line); - } - return TRUE; + Modify(true); + + event.Skip(); } -#endif -void DoodleSegment::Draw(wxDC *dc) +// ---------------------------------------------------------------------------- +// TextEditDocument implementation +// ---------------------------------------------------------------------------- + +IMPLEMENT_DYNAMIC_CLASS(TextEditDocument, wxDocument) + +wxTextCtrl* TextEditDocument::GetTextCtrl() const { - wxNode *node = lines.First(); - while (node) - { - DoodleLine *line = (DoodleLine *)node->Data(); - dc->DrawLine(line->x1, line->y1, line->x2, line->y2); - node = node->Next(); - } + wxView* view = GetFirstView(); + return view ? wxStaticCast(view, TextEditView)->GetText() : NULL; } -/* - * Implementation of drawing command - */ +// ---------------------------------------------------------------------------- +// wxImageDocument implementation +// ---------------------------------------------------------------------------- + +///////////////////////////////////////////////////////////////////////////// +// wxImageDocument -DrawingCommand::DrawingCommand(const wxString& name, int command, DrawingDocument *ddoc, DoodleSegment *seg): - wxCommand(TRUE, name) +IMPLEMENT_DYNAMIC_CLASS(wxImageDocument, wxDocument) + +wxImageDocument::wxImageDocument() : wxDocument() { - doc = ddoc; - segment = seg; - cmd = command; } -DrawingCommand::~DrawingCommand(void) +wxImageDocument::~wxImageDocument() { - if (segment) - delete segment; } -bool DrawingCommand::Do(void) +bool wxImageDocument::DeleteContents() { - switch (cmd) - { - case DOODLE_CUT: - { - // Cut the last segment - if (doc->GetDoodleSegments().Number() > 0) - { - wxNode *node = doc->GetDoodleSegments().Last(); - if (segment) - delete segment; - - segment = (DoodleSegment *)node->Data(); - delete node; - - doc->Modify(TRUE); - doc->UpdateAllViews(); - } - break; - } - case DOODLE_ADD: + bool ok = wxDocument::DeleteContents(); + if (ok && m_image.IsOk()) { - doc->GetDoodleSegments().Append(new DoodleSegment(*segment)); - doc->Modify(TRUE); - doc->UpdateAllViews(); - break; + m_image.Destroy(); } - } - return TRUE; + return ok; } -bool DrawingCommand::Undo(void) +bool wxImageDocument::SaveFile(wxOutputStream* stream, wxBitmapType type) const { - switch (cmd) - { - case DOODLE_CUT: - { - // Paste the segment - if (segment) - { - doc->GetDoodleSegments().Append(segment); - doc->Modify(TRUE); - doc->UpdateAllViews(); - segment = (DoodleSegment *) NULL; - } - doc->Modify(TRUE); - doc->UpdateAllViews(); - break; - } - case DOODLE_ADD: - { - // Cut the last segment - if (doc->GetDoodleSegments().Number() > 0) - { - wxNode *node = doc->GetDoodleSegments().Last(); - DoodleSegment *seg = (DoodleSegment *)node->Data(); - delete seg; - delete node; - - doc->Modify(TRUE); - doc->UpdateAllViews(); - } - } - } - return TRUE; + return m_image.IsOk() && m_image.SaveFile(*stream, type); } -IMPLEMENT_DYNAMIC_CLASS(TextEditDocument, wxDocument) - -// Since text windows have their own method for saving to/loading from files, -// we override OnSave/OpenDocument instead of Save/LoadObject -bool TextEditDocument::OnSaveDocument(const wxString& filename) +bool wxImageDocument::DoOpenDocument(const wxString& file) { - TextEditView *view = (TextEditView *)GetFirstView(); - - if (!view->textsw->SaveFile(filename)) - return FALSE; - Modify(FALSE); - return TRUE; + wxFileInputStream stream(file); + return stream.IsOk() && DoOpenDocument(&stream); } -bool TextEditDocument::OnOpenDocument(const wxString& filename) +bool wxImageDocument::DoSaveDocument(const wxString& file) { - TextEditView *view = (TextEditView *)GetFirstView(); - if (!view->textsw->LoadFile(filename)) - return FALSE; - - SetFilename(filename, TRUE); - Modify(FALSE); - UpdateAllViews(); - return TRUE; + wxFileOutputStream stream(file); + return stream.IsOk() && DoSaveDocument(&stream); } -bool TextEditDocument::IsModified(void) const +bool wxImageDocument::DoOpenDocument(wxInputStream* stream) { - TextEditView *view = (TextEditView *)GetFirstView(); - if (view) - { - return (wxDocument::IsModified() || view->textsw->IsModified()); - } - else - return wxDocument::IsModified(); + return m_image.LoadFile(*stream); } -void TextEditDocument::Modify(bool mod) +bool wxImageDocument::DoSaveDocument(wxOutputStream* stream) const { - TextEditView *view = (TextEditView *)GetFirstView(); - - wxDocument::Modify(mod); - - if (!mod && view && view->textsw) - view->textsw->DiscardEdits(); + return m_image.IsOk() && SaveFile(stream, m_image.GetType()); } +