]> git.saurik.com Git - wxWidgets.git/blame - contrib/samples/ogl/ogledit/doc.cpp
Force size to integer coordinates in all cases.
[wxWidgets.git] / contrib / samples / ogl / ogledit / doc.cpp
CommitLineData
1fc25a89
JS
1/////////////////////////////////////////////////////////////////////////////
2// Name: doc.cpp
3// Purpose: Implements document functionality in OGLEdit
4// Author: Julian Smart
5// Modified by:
6// Created: 12/07/98
7// RCS-ID: $Id$
8// Copyright: (c) Julian Smart
2ba06d5a 9// Licence: wxWindows licence
1fc25a89
JS
10/////////////////////////////////////////////////////////////////////////////
11
12#ifdef __GNUG__
13// #pragma implementation
14#endif
15
16// For compilers that support precompilation, includes "wx.h".
92a19c2e 17#include "wx/wxprec.h"
1fc25a89
JS
18
19#ifdef __BORLANDC__
20#pragma hdrstop
21#endif
22
23#ifndef WX_PRECOMP
24#include <wx/wx.h>
25#endif
26
27#if !wxUSE_DOC_VIEW_ARCHITECTURE
28#error You must set wxUSE_DOC_VIEW_ARCHITECTURE to 1 in wx_setup.h!
29#endif
30
1fc25a89
JS
31#include "ogledit.h"
32#include "doc.h"
33#include "view.h"
34
35#if wxUSE_STD_IOSTREAM
dd107c50 36#include <ioswrap.h>
1fc25a89
JS
37#endif
38
39IMPLEMENT_DYNAMIC_CLASS(DiagramDocument, wxDocument)
40
41DiagramDocument::DiagramDocument(void)
42{
43}
44
45DiagramDocument::~DiagramDocument(void)
46{
47}
48
49bool DiagramDocument::OnCloseDocument(void)
50{
51 diagram.DeleteAllShapes();
2ba06d5a 52 return true;
1fc25a89
JS
53}
54
55#if wxUSE_STD_IOSTREAM
dd107c50 56wxSTD ostream& DiagramDocument::SaveObject(wxSTD ostream& stream)
1fc25a89
JS
57{
58 wxDocument::SaveObject(stream);
55c91e8a 59
1fc25a89
JS
60 char buf[400];
61 (void) wxGetTempFileName("diag", buf);
62
63 diagram.SaveFile(buf);
64 wxTransferFileToStream(buf, stream);
65
66 wxRemoveFile(buf);
55c91e8a 67
1fc25a89
JS
68 return stream;
69}
70
dd107c50 71wxSTD istream& DiagramDocument::LoadObject(wxSTD istream& stream)
1fc25a89
JS
72{
73 wxDocument::LoadObject(stream);
74
75 char buf[400];
76 (void) wxGetTempFileName("diag", buf);
77
78 wxTransferStreamToFile(stream, buf);
79
80 diagram.DeleteAllShapes();
81 diagram.LoadFile(buf);
82 wxRemoveFile(buf);
83
84 return stream;
85}
86#else
87
88wxOutputStream& DiagramDocument::SaveObject(wxOutputStream& stream)
89{
1dde66dd
JS
90#if wxUSE_PROLOGIO
91
1fc25a89 92 wxDocument::SaveObject(stream);
1484b5cc
VS
93 wxChar buf[400];
94 (void) wxGetTempFileName(_T("diag"), buf);
1fc25a89
JS
95
96 diagram.SaveFile(buf);
97
98 wxTransferFileToStream(buf, stream);
99
100 wxRemoveFile(buf);
1dde66dd
JS
101
102#endif
1fc25a89
JS
103
104 return stream;
105}
106
107wxInputStream& DiagramDocument::LoadObject(wxInputStream& stream)
108{
1dde66dd 109#if wxUSE_PROLOGIO
1fc25a89
JS
110 wxDocument::LoadObject(stream);
111
1484b5cc
VS
112 wxChar buf[400];
113 (void) wxGetTempFileName(_T("diag"), buf);
1fc25a89
JS
114
115 wxTransferStreamToFile(stream, buf);
116
117 diagram.DeleteAllShapes();
118 diagram.LoadFile(buf);
119 wxRemoveFile(buf);
1dde66dd 120#endif
1fc25a89
JS
121
122 return stream;
123}
124
125#endif
126
127/*
128 * Implementation of drawing command
129 */
130
1484b5cc 131DiagramCommand::DiagramCommand(const wxString& name, int command, DiagramDocument *ddoc, wxClassInfo *info, double xx, double yy,
1fc25a89 132 bool sel, wxShape *theShape, wxShape *fs, wxShape *ts):
2ba06d5a 133 wxCommand(true, name)
1fc25a89
JS
134{
135 doc = ddoc;
136 cmd = command;
137 shape = theShape;
138 fromShape = fs;
139 toShape = ts;
140 shapeInfo = info;
141 shapeBrush = NULL;
142 shapePen = NULL;
143 x = xx;
144 y = yy;
145 selected = sel;
2ba06d5a 146 deleteShape = false;
1fc25a89
JS
147}
148
1484b5cc 149DiagramCommand::DiagramCommand(const wxString& name, int command, DiagramDocument *ddoc, wxBrush *backgroundColour, wxShape *theShape):
2ba06d5a 150 wxCommand(true, name)
1fc25a89
JS
151{
152 doc = ddoc;
153 cmd = command;
154 shape = theShape;
155 fromShape = NULL;
156 toShape = NULL;
157 shapeInfo = NULL;
158 x = 0.0;
159 y = 0.0;
2ba06d5a
WS
160 selected = false;
161 deleteShape = false;
1fc25a89
JS
162 shapeBrush = backgroundColour;
163 shapePen = NULL;
164}
165
1484b5cc 166DiagramCommand::DiagramCommand(const wxString& name, int command, DiagramDocument *ddoc, const wxString& lab, wxShape *theShape):
2ba06d5a 167 wxCommand(true, name)
1fc25a89
JS
168{
169 doc = ddoc;
170 cmd = command;
171 shape = theShape;
172 fromShape = NULL;
173 toShape = NULL;
174 shapeInfo = NULL;
175 x = 0.0;
176 y = 0.0;
2ba06d5a
WS
177 selected = false;
178 deleteShape = false;
1fc25a89
JS
179 shapeBrush = NULL;
180 shapePen = NULL;
181 shapeLabel = lab;
182}
183
184DiagramCommand::~DiagramCommand(void)
185{
186 if (shape && deleteShape)
187 {
188 shape->SetCanvas(NULL);
189 delete shape;
190 }
191}
192
193bool DiagramCommand::Do(void)
194{
195 switch (cmd)
196 {
cecdcad1 197 case wxID_CUT:
1fc25a89
JS
198 {
199 if (shape)
200 {
2ba06d5a 201 deleteShape = true;
55c91e8a 202
2ba06d5a 203 shape->Select(false);
55c91e8a 204
1fc25a89
JS
205 // Generate commands to explicitly remove each connected line.
206 RemoveLines(shape);
55c91e8a 207
1fc25a89
JS
208 doc->GetDiagram()->RemoveShape(shape);
209 if (shape->IsKindOf(CLASSINFO(wxLineShape)))
210 {
211 wxLineShape *lineShape = (wxLineShape *)shape;
212 fromShape = lineShape->GetFrom();
213 toShape = lineShape->GetTo();
214 }
215 shape->Unlink();
55c91e8a 216
2ba06d5a 217 doc->Modify(true);
1fc25a89
JS
218 doc->UpdateAllViews();
219 }
220
221 break;
222 }
223 case OGLEDIT_ADD_SHAPE:
224 {
8552e6f0 225 wxShape *theShape;
1fc25a89
JS
226 if (shape)
227 theShape = shape; // Saved from undoing the shape
228 else
229 {
230 theShape = (wxShape *)shapeInfo->CreateObject();
231 theShape->AssignNewIds();
1484b5cc 232 theShape->SetEventHandler(new MyEvtHandler(theShape, theShape, wxEmptyString));
2ba06d5a 233 theShape->SetCentreResize(false);
1fc25a89
JS
234 theShape->SetPen(wxBLACK_PEN);
235 theShape->SetBrush(wxCYAN_BRUSH);
55c91e8a 236
1fc25a89
JS
237 theShape->SetSize(60, 60);
238 }
239 doc->GetDiagram()->AddShape(theShape);
2ba06d5a 240 theShape->Show(true);
1fc25a89
JS
241
242 wxClientDC dc(theShape->GetCanvas());
243 theShape->GetCanvas()->PrepareDC(dc);
244
245 theShape->Move(dc, x, y);
55c91e8a 246
1fc25a89 247 shape = theShape;
2ba06d5a 248 deleteShape = false;
1fc25a89 249
2ba06d5a 250 doc->Modify(true);
1fc25a89
JS
251 doc->UpdateAllViews();
252 break;
253 }
254 case OGLEDIT_ADD_LINE:
255 {
8552e6f0 256 wxShape *theShape;
1fc25a89
JS
257 if (shape)
258 theShape = shape; // Saved from undoing the line
259 else
260 {
261 theShape = (wxShape *)shapeInfo->CreateObject();
262 theShape->AssignNewIds();
1484b5cc 263 theShape->SetEventHandler(new MyEvtHandler(theShape, theShape, wxEmptyString));
1fc25a89
JS
264 theShape->SetPen(wxBLACK_PEN);
265 theShape->SetBrush(wxRED_BRUSH);
266
267 wxLineShape *lineShape = (wxLineShape *)theShape;
268
269 // Yes, you can have more than 2 control points, in which case
270 // it becomes a multi-segment line.
271 lineShape->MakeLineControlPoints(2);
1484b5cc 272 lineShape->AddArrow(ARROW_ARROW, ARROW_POSITION_END, 10.0, 0.0, _T("Normal arrowhead"));
1fc25a89 273 }
55c91e8a 274
1fc25a89 275 doc->GetDiagram()->AddShape(theShape);
55c91e8a 276
1fc25a89 277 fromShape->AddLine((wxLineShape *)theShape, toShape);
55c91e8a 278
2ba06d5a 279 theShape->Show(true);
1fc25a89
JS
280
281 wxClientDC dc(theShape->GetCanvas());
282 theShape->GetCanvas()->PrepareDC(dc);
283
284 // It won't get drawn properly unless you move both
285 // connected images
286 fromShape->Move(dc, fromShape->GetX(), fromShape->GetY());
287 toShape->Move(dc, toShape->GetX(), toShape->GetY());
55c91e8a 288
1fc25a89 289 shape = theShape;
2ba06d5a 290 deleteShape = false;
1fc25a89 291
2ba06d5a 292 doc->Modify(true);
1fc25a89
JS
293 doc->UpdateAllViews();
294 break;
295 }
296 case OGLEDIT_CHANGE_BACKGROUND_COLOUR:
297 {
298 if (shape)
299 {
300 wxClientDC dc(shape->GetCanvas());
301 shape->GetCanvas()->PrepareDC(dc);
302
303 wxBrush *oldBrush = shape->GetBrush();
304 shape->SetBrush(shapeBrush);
305 shapeBrush = oldBrush;
306 shape->Draw(dc);
55c91e8a 307
2ba06d5a 308 doc->Modify(true);
1fc25a89
JS
309 doc->UpdateAllViews();
310 }
311
312 break;
313 }
314 case OGLEDIT_EDIT_LABEL:
315 {
316 if (shape)
317 {
318 MyEvtHandler *myHandler = (MyEvtHandler *)shape->GetEventHandler();
319 wxString oldLabel(myHandler->label);
320 myHandler->label = shapeLabel;
321 shapeLabel = oldLabel;
322
323 wxClientDC dc(shape->GetCanvas());
324 shape->GetCanvas()->PrepareDC(dc);
325
1484b5cc 326 shape->FormatText(dc, /* (char*) (const char*) */ myHandler->label);
1fc25a89 327 shape->Draw(dc);
55c91e8a 328
2ba06d5a 329 doc->Modify(true);
1fc25a89
JS
330 doc->UpdateAllViews();
331 }
332
333 break;
334 }
335 }
2ba06d5a 336 return true;
1fc25a89
JS
337}
338
339bool DiagramCommand::Undo(void)
340{
341 switch (cmd)
342 {
cecdcad1 343 case wxID_CUT:
1fc25a89
JS
344 {
345 if (shape)
346 {
347 doc->GetDiagram()->AddShape(shape);
2ba06d5a 348 shape->Show(true);
1fc25a89
JS
349
350 if (shape->IsKindOf(CLASSINFO(wxLineShape)))
351 {
352 wxLineShape *lineShape = (wxLineShape *)shape;
353
354 fromShape->AddLine(lineShape, toShape);
355 }
356 if (selected)
2ba06d5a 357 shape->Select(true);
1fc25a89 358
2ba06d5a 359 deleteShape = false;
1fc25a89 360 }
2ba06d5a 361 doc->Modify(true);
1fc25a89
JS
362 doc->UpdateAllViews();
363 break;
364 }
365 case OGLEDIT_ADD_SHAPE:
366 case OGLEDIT_ADD_LINE:
367 {
368 if (shape)
369 {
370 wxClientDC dc(shape->GetCanvas());
371 shape->GetCanvas()->PrepareDC(dc);
372
2ba06d5a 373 shape->Select(false, &dc);
1fc25a89
JS
374 doc->GetDiagram()->RemoveShape(shape);
375 shape->Unlink();
2ba06d5a 376 deleteShape = true;
1fc25a89 377 }
2ba06d5a 378 doc->Modify(true);
1fc25a89
JS
379 doc->UpdateAllViews();
380 break;
381 }
382 case OGLEDIT_CHANGE_BACKGROUND_COLOUR:
383 {
384 if (shape)
385 {
386 wxClientDC dc(shape->GetCanvas());
387 shape->GetCanvas()->PrepareDC(dc);
388
389 wxBrush *oldBrush = shape->GetBrush();
390 shape->SetBrush(shapeBrush);
391 shapeBrush = oldBrush;
392 shape->Draw(dc);
55c91e8a 393
2ba06d5a 394 doc->Modify(true);
1fc25a89
JS
395 doc->UpdateAllViews();
396 }
397 break;
398 }
399 case OGLEDIT_EDIT_LABEL:
400 {
401 if (shape)
402 {
403 MyEvtHandler *myHandler = (MyEvtHandler *)shape->GetEventHandler();
404 wxString oldLabel(myHandler->label);
405 myHandler->label = shapeLabel;
406 shapeLabel = oldLabel;
407
408 wxClientDC dc(shape->GetCanvas());
409 shape->GetCanvas()->PrepareDC(dc);
410
1484b5cc 411 shape->FormatText(dc, /* (char*) (const char*) */ myHandler->label);
1fc25a89 412 shape->Draw(dc);
55c91e8a 413
2ba06d5a 414 doc->Modify(true);
1fc25a89
JS
415 doc->UpdateAllViews();
416 }
417
418 break;
419 }
420 }
2ba06d5a 421 return true;
1fc25a89
JS
422}
423
424// Remove each individual line connected to a shape by sending a command.
425void DiagramCommand::RemoveLines(wxShape *shape)
426{
36ca94a2 427 wxObjectList::compatibility_iterator node = shape->GetLines().GetFirst();
1fc25a89
JS
428 while (node)
429 {
8552e6f0 430 wxLineShape *line = (wxLineShape *)node->GetData();
cecdcad1 431 doc->GetCommandProcessor()->Submit(new DiagramCommand(_T("Cut"), wxID_CUT, doc, NULL, 0.0, 0.0, line->Selected(), line));
55c91e8a 432
8552e6f0 433 node = shape->GetLines().GetFirst();
1fc25a89
JS
434 }
435}
436
437/*
438 * MyEvtHandler: an event handler class for all shapes
439 */
55c91e8a 440
1484b5cc 441void MyEvtHandler::OnLeftClick(double WXUNUSED(x), double WXUNUSED(y), int keys, int WXUNUSED(attachment))
1fc25a89
JS
442{
443 wxClientDC dc(GetShape()->GetCanvas());
444 GetShape()->GetCanvas()->PrepareDC(dc);
445
446 if (keys == 0)
447 {
448 // Selection is a concept the library knows about
449 if (GetShape()->Selected())
450 {
2ba06d5a 451 GetShape()->Select(false, &dc);
1fc25a89
JS
452 GetShape()->GetCanvas()->Redraw(dc); // Redraw because bits of objects will be are missing
453 }
454 else
455 {
456 // Ensure no other shape is selected, to simplify Undo/Redo code
2ba06d5a 457 bool redraw = false;
36ca94a2 458 wxObjectList::compatibility_iterator node = GetShape()->GetCanvas()->GetDiagram()->GetShapeList()->GetFirst();
1fc25a89
JS
459 while (node)
460 {
8552e6f0 461 wxShape *eachShape = (wxShape *)node->GetData();
1fc25a89
JS
462 if (eachShape->GetParent() == NULL)
463 {
464 if (eachShape->Selected())
465 {
2ba06d5a
WS
466 eachShape->Select(false, &dc);
467 redraw = true;
1fc25a89
JS
468 }
469 }
8552e6f0 470 node = node->GetNext();
1fc25a89 471 }
2ba06d5a 472 GetShape()->Select(true, &dc);
1fc25a89
JS
473 if (redraw)
474 GetShape()->GetCanvas()->Redraw(dc);
475 }
476 }
477 else if (keys & KEY_CTRL)
478 {
479 // Do something for CONTROL
480 }
481 else
482 {
d96cdd4a 483#if wxUSE_STATUSBAR
1fc25a89 484 wxGetApp().frame->SetStatusText(label);
d96cdd4a 485#endif // wxUSE_STATUSBAR
1fc25a89
JS
486 }
487}
488
489/*
490 * Implement connection of two shapes by right-dragging between them.
491 */
492
1484b5cc 493void MyEvtHandler::OnBeginDragRight(double x, double y, int WXUNUSED(keys), int attachment)
1fc25a89
JS
494{
495 // Force attachment to be zero for now. Eventually we can deal with
496 // the actual attachment point, e.g. a rectangle side if attachment mode is on.
497 attachment = 0;
55c91e8a 498
1fc25a89
JS
499 wxClientDC dc(GetShape()->GetCanvas());
500 GetShape()->GetCanvas()->PrepareDC(dc);
501
55c91e8a 502 wxPen dottedPen(*wxBLACK, 1, wxDOT);
1fc25a89
JS
503 dc.SetLogicalFunction(OGLRBLF);
504 dc.SetPen(dottedPen);
505 double xp, yp;
506 GetShape()->GetAttachmentPosition(attachment, &xp, &yp);
507 dc.DrawLine((long) xp, (long) yp, (long) x, (long) y);
508 GetShape()->GetCanvas()->CaptureMouse();
509}
510
1484b5cc 511void MyEvtHandler::OnDragRight(bool WXUNUSED(draw), double x, double y, int WXUNUSED(keys), int attachment)
1fc25a89
JS
512{
513 // Force attachment to be zero for now
514 attachment = 0;
515
516 wxClientDC dc(GetShape()->GetCanvas());
517 GetShape()->GetCanvas()->PrepareDC(dc);
518
55c91e8a 519 wxPen dottedPen(*wxBLACK, 1, wxDOT);
1fc25a89
JS
520 dc.SetLogicalFunction(OGLRBLF);
521 dc.SetPen(dottedPen);
522 double xp, yp;
523 GetShape()->GetAttachmentPosition(attachment, &xp, &yp);
524 dc.DrawLine((long) xp, (long) yp, (long) x, (long) y);
525}
526
1484b5cc 527void MyEvtHandler::OnEndDragRight(double x, double y, int WXUNUSED(keys), int WXUNUSED(attachment))
1fc25a89
JS
528{
529 GetShape()->GetCanvas()->ReleaseMouse();
530 MyCanvas *canvas = (MyCanvas *)GetShape()->GetCanvas();
531
532 // Check if we're on an object
533 int new_attachment;
534 wxShape *otherShape = canvas->FindFirstSensitiveShape(x, y, &new_attachment, OP_DRAG_RIGHT);
55c91e8a 535
1fc25a89
JS
536 if (otherShape && !otherShape->IsKindOf(CLASSINFO(wxLineShape)))
537 {
538 canvas->view->GetDocument()->GetCommandProcessor()->Submit(
1484b5cc 539 new DiagramCommand(_T("wxLineShape"), OGLEDIT_ADD_LINE, (DiagramDocument *)canvas->view->GetDocument(), CLASSINFO(wxLineShape),
2ba06d5a 540 0.0, 0.0, false, NULL, GetShape(), otherShape));
1fc25a89
JS
541 }
542}
543
1484b5cc 544void MyEvtHandler::OnEndSize(double WXUNUSED(x), double WXUNUSED(y))
1fc25a89
JS
545{
546 wxClientDC dc(GetShape()->GetCanvas());
547 GetShape()->GetCanvas()->PrepareDC(dc);
548
1484b5cc 549 GetShape()->FormatText(dc, /* (char*) (const char*) */ label);
1fc25a89
JS
550}
551
552/*
553 * Diagram
554 */
1dde66dd
JS
555
556#if wxUSE_PROLOGIO
55c91e8a 557
1fc25a89
JS
558bool MyDiagram::OnShapeSave(wxExprDatabase& db, wxShape& shape, wxExpr& expr)
559{
560 wxDiagram::OnShapeSave(db, shape, expr);
561 MyEvtHandler *handler = (MyEvtHandler *)shape.GetEventHandler();
1484b5cc 562 expr.AddAttributeValueString(_T("label"), handler->label);
2ba06d5a 563 return true;
1fc25a89
JS
564}
565
566bool MyDiagram::OnShapeLoad(wxExprDatabase& db, wxShape& shape, wxExpr& expr)
567{
568 wxDiagram::OnShapeLoad(db, shape, expr);
1484b5cc
VS
569 wxChar *label = NULL;
570 expr.AssignAttributeValue(_T("label"), &label);
1fc25a89
JS
571 MyEvtHandler *handler = new MyEvtHandler(&shape, &shape, wxString(label));
572 shape.SetEventHandler(handler);
55c91e8a 573
1fc25a89
JS
574 if (label)
575 delete[] label;
2ba06d5a 576 return true;
1fc25a89
JS
577}
578
1dde66dd
JS
579#endif
580
1fc25a89
JS
581/*
582 * New shapes
583 */
584
585IMPLEMENT_DYNAMIC_CLASS(wxRoundedRectangleShape, wxRectangleShape)
586
587wxRoundedRectangleShape::wxRoundedRectangleShape(double w, double h):
588 wxRectangleShape(w, h)
589{
590 // 0.3 of the smaller rectangle dimension
591 SetCornerRadius((double) -0.3);
592}
593
594IMPLEMENT_DYNAMIC_CLASS(wxDiamondShape, wxPolygonShape)
595
596wxDiamondShape::wxDiamondShape(double w, double h):
597 wxPolygonShape()
598{
599 // wxPolygonShape::SetSize relies on the shape having non-zero
600 // size initially.
601 if (w == 0.0)
602 w = 60.0;
603 if (h == 0.0)
604 h = 60.0;
55c91e8a 605
1fc25a89
JS
606 wxList *thePoints = new wxList;
607 wxRealPoint *point = new wxRealPoint(0.0, (-h/2.0));
608 thePoints->Append((wxObject*) point);
609
610 point = new wxRealPoint((w/2.0), 0.0);
611 thePoints->Append((wxObject*) point);
612
613 point = new wxRealPoint(0.0, (h/2.0));
614 thePoints->Append((wxObject*) point);
615
616 point = new wxRealPoint((-w/2.0), 0.0);
617 thePoints->Append((wxObject*) point);
618
619 Create(thePoints);
620}