]> git.saurik.com Git - wxWidgets.git/blame - src/richtext/richtextbuffer.cpp
Recognize the numpad cursor keys too
[wxWidgets.git] / src / richtext / richtextbuffer.cpp
CommitLineData
5d7836c4 1/////////////////////////////////////////////////////////////////////////////
61399247 2// Name: src/richtext/richtextbuffer.cpp
5d7836c4
JS
3// Purpose: Buffer for wxRichTextCtrl
4// Author: Julian Smart
7fe8059f 5// Modified by:
5d7836c4 6// Created: 2005-09-30
7fe8059f 7// RCS-ID: $Id$
5d7836c4
JS
8// Copyright: (c) Julian Smart
9// Licence: wxWindows licence
10/////////////////////////////////////////////////////////////////////////////
11
12// For compilers that support precompilation, includes "wx.h".
13#include "wx/wxprec.h"
14
15#ifdef __BORLANDC__
61399247 16 #pragma hdrstop
5d7836c4
JS
17#endif
18
b01ca8b6
JS
19#if wxUSE_RICHTEXT
20
21#include "wx/richtext/richtextbuffer.h"
22
5d7836c4 23#ifndef WX_PRECOMP
61399247
WS
24 #include "wx/dc.h"
25 #include "wx/intl.h"
7947a48a 26 #include "wx/log.h"
28f92d74 27 #include "wx/dataobj.h"
5d7836c4
JS
28#endif
29
5d7836c4
JS
30#include "wx/filename.h"
31#include "wx/clipbrd.h"
32#include "wx/wfstream.h"
33#include "wx/module.h"
34#include "wx/mstream.h"
35#include "wx/sstream.h"
36
5d7836c4
JS
37#include "wx/richtext/richtextctrl.h"
38#include "wx/richtext/richtextstyles.h"
39
40#include "wx/listimpl.cpp"
41
412e0d47
DS
42WX_DEFINE_LIST(wxRichTextObjectList)
43WX_DEFINE_LIST(wxRichTextLineList)
5d7836c4
JS
44
45/*!
46 * wxRichTextObject
47 * This is the base for drawable objects.
48 */
49
50IMPLEMENT_CLASS(wxRichTextObject, wxObject)
51
52wxRichTextObject::wxRichTextObject(wxRichTextObject* parent)
53{
54 m_dirty = false;
55 m_refCount = 1;
56 m_parent = parent;
57 m_leftMargin = 0;
58 m_rightMargin = 0;
59 m_topMargin = 0;
60 m_bottomMargin = 0;
61 m_descent = 0;
62}
63
64wxRichTextObject::~wxRichTextObject()
65{
66}
67
68void wxRichTextObject::Dereference()
69{
70 m_refCount --;
71 if (m_refCount <= 0)
72 delete this;
73}
74
75/// Copy
76void wxRichTextObject::Copy(const wxRichTextObject& obj)
77{
78 m_size = obj.m_size;
79 m_pos = obj.m_pos;
80 m_dirty = obj.m_dirty;
81 m_range = obj.m_range;
82 m_attributes = obj.m_attributes;
83 m_descent = obj.m_descent;
84
85 if (!m_attributes.GetFont().Ok())
86 wxLogDebug(wxT("No font!"));
87 if (!obj.m_attributes.GetFont().Ok())
88 wxLogDebug(wxT("Parent has no font!"));
89}
90
91void wxRichTextObject::SetMargins(int margin)
92{
93 m_leftMargin = m_rightMargin = m_topMargin = m_bottomMargin = margin;
94}
95
96void wxRichTextObject::SetMargins(int leftMargin, int rightMargin, int topMargin, int bottomMargin)
97{
98 m_leftMargin = leftMargin;
99 m_rightMargin = rightMargin;
100 m_topMargin = topMargin;
101 m_bottomMargin = bottomMargin;
102}
103
104// Convert units in tends of a millimetre to device units
105int wxRichTextObject::ConvertTenthsMMToPixels(wxDC& dc, int units)
106{
107 int ppi = dc.GetPPI().x;
108
109 // There are ppi pixels in 254.1 "1/10 mm"
110
111 double pixels = ((double) units * (double)ppi) / 254.1;
112
113 return (int) pixels;
114}
115
116/// Dump to output stream for debugging
117void wxRichTextObject::Dump(wxTextOutputStream& stream)
118{
119 stream << GetClassInfo()->GetClassName() << wxT("\n");
120 stream << wxString::Format(wxT("Size: %d,%d. Position: %d,%d, Range: %ld,%ld"), m_size.x, m_size.y, m_pos.x, m_pos.y, m_range.GetStart(), m_range.GetEnd()) << wxT("\n");
121 stream << wxString::Format(wxT("Text colour: %d,%d,%d."), (int) m_attributes.GetTextColour().Red(), (int) m_attributes.GetTextColour().Green(), (int) m_attributes.GetTextColour().Blue()) << wxT("\n");
122}
123
124
125/*!
126 * wxRichTextCompositeObject
127 * This is the base for drawable objects.
128 */
129
130IMPLEMENT_CLASS(wxRichTextCompositeObject, wxRichTextObject)
131
132wxRichTextCompositeObject::wxRichTextCompositeObject(wxRichTextObject* parent):
133 wxRichTextObject(parent)
134{
135}
136
137wxRichTextCompositeObject::~wxRichTextCompositeObject()
138{
139 DeleteChildren();
140}
141
142/// Get the nth child
143wxRichTextObject* wxRichTextCompositeObject::GetChild(size_t n) const
144{
145 wxASSERT ( n < m_children.GetCount() );
146
147 return m_children.Item(n)->GetData();
148}
149
150/// Append a child, returning the position
151size_t wxRichTextCompositeObject::AppendChild(wxRichTextObject* child)
152{
153 m_children.Append(child);
154 child->SetParent(this);
155 return m_children.GetCount() - 1;
156}
157
158/// Insert the child in front of the given object, or at the beginning
159bool wxRichTextCompositeObject::InsertChild(wxRichTextObject* child, wxRichTextObject* inFrontOf)
160{
161 if (inFrontOf)
162 {
163 wxRichTextObjectList::compatibility_iterator node = m_children.Find(inFrontOf);
164 m_children.Insert(node, child);
165 }
166 else
167 m_children.Insert(child);
168 child->SetParent(this);
169
170 return true;
171}
172
173/// Delete the child
174bool wxRichTextCompositeObject::RemoveChild(wxRichTextObject* child, bool deleteChild)
175{
176 wxRichTextObjectList::compatibility_iterator node = m_children.Find(child);
177 if (node)
178 {
efbf6735
JS
179 wxRichTextObject* obj = node->GetData();
180 m_children.Erase(node);
5d7836c4 181 if (deleteChild)
efbf6735 182 delete obj;
5d7836c4
JS
183
184 return true;
185 }
186 return false;
187}
188
189/// Delete all children
190bool wxRichTextCompositeObject::DeleteChildren()
191{
192 wxRichTextObjectList::compatibility_iterator node = m_children.GetFirst();
193 while (node)
194 {
195 wxRichTextObjectList::compatibility_iterator oldNode = node;
196
197 wxRichTextObject* child = node->GetData();
198 child->Dereference(); // Only delete if reference count is zero
199
200 node = node->GetNext();
efbf6735 201 m_children.Erase(oldNode);
5d7836c4
JS
202 }
203
204 return true;
205}
206
207/// Get the child count
208size_t wxRichTextCompositeObject::GetChildCount() const
209{
210 return m_children.GetCount();
211}
212
213/// Copy
214void wxRichTextCompositeObject::Copy(const wxRichTextCompositeObject& obj)
215{
216 wxRichTextObject::Copy(obj);
217
218 DeleteChildren();
219
220 wxRichTextObjectList::compatibility_iterator node = obj.m_children.GetFirst();
221 while (node)
222 {
223 wxRichTextObject* child = node->GetData();
224 m_children.Append(child->Clone());
225
226 node = node->GetNext();
227 }
228}
229
230/// Hit-testing: returns a flag indicating hit test details, plus
231/// information about position
232int wxRichTextCompositeObject::HitTest(wxDC& dc, const wxPoint& pt, long& textPosition)
233{
234 wxRichTextObjectList::compatibility_iterator node = m_children.GetFirst();
235 while (node)
236 {
237 wxRichTextObject* child = node->GetData();
238
239 int ret = child->HitTest(dc, pt, textPosition);
240 if (ret != wxRICHTEXT_HITTEST_NONE)
241 return ret;
242
243 node = node->GetNext();
244 }
245
246 return wxRICHTEXT_HITTEST_NONE;
247}
248
249/// Finds the absolute position and row height for the given character position
250bool wxRichTextCompositeObject::FindPosition(wxDC& dc, long index, wxPoint& pt, int* height, bool forceLineStart)
251{
252 wxRichTextObjectList::compatibility_iterator node = m_children.GetFirst();
253 while (node)
254 {
255 wxRichTextObject* child = node->GetData();
256
257 if (child->FindPosition(dc, index, pt, height, forceLineStart))
258 return true;
259
260 node = node->GetNext();
261 }
262
263 return false;
264}
265
266/// Calculate range
267void wxRichTextCompositeObject::CalculateRange(long start, long& end)
268{
269 long current = start;
270 long lastEnd = current;
271
272 wxRichTextObjectList::compatibility_iterator node = m_children.GetFirst();
273 while (node)
274 {
275 wxRichTextObject* child = node->GetData();
276 long childEnd = 0;
277
278 child->CalculateRange(current, childEnd);
279 lastEnd = childEnd;
280
281 current = childEnd + 1;
282
283 node = node->GetNext();
284 }
285
286 end = lastEnd;
287
288 // An object with no children has zero length
289 if (m_children.GetCount() == 0)
290 end --;
291
292 m_range.SetRange(start, end);
293}
294
295/// Delete range from layout.
296bool wxRichTextCompositeObject::DeleteRange(const wxRichTextRange& range)
297{
298 wxRichTextObjectList::compatibility_iterator node = m_children.GetFirst();
7fe8059f 299
5d7836c4
JS
300 while (node)
301 {
302 wxRichTextObject* obj = (wxRichTextObject*) node->GetData();
303 wxRichTextObjectList::compatibility_iterator next = node->GetNext();
7fe8059f 304
5d7836c4
JS
305 // Delete the range in each paragraph
306
307 // When a chunk has been deleted, internally the content does not
308 // now match the ranges.
309 // However, so long as deletion is not done on the same object twice this is OK.
310 // If you may delete content from the same object twice, recalculate
311 // the ranges inbetween DeleteRange calls by calling CalculateRanges, and
312 // adjust the range you're deleting accordingly.
7fe8059f 313
5d7836c4
JS
314 if (!obj->GetRange().IsOutside(range))
315 {
316 obj->DeleteRange(range);
317
318 // Delete an empty object, or paragraph within this range.
319 if (obj->IsEmpty() ||
320 (range.GetStart() <= obj->GetRange().GetStart() && range.GetEnd() >= obj->GetRange().GetEnd()))
321 {
322 // An empty paragraph has length 1, so won't be deleted unless the
323 // whole range is deleted.
7fe8059f 324 RemoveChild(obj, true);
5d7836c4
JS
325 }
326 }
7fe8059f 327
5d7836c4
JS
328 node = next;
329 }
7fe8059f 330
5d7836c4
JS
331 return true;
332}
333
334/// Get any text in this object for the given range
335wxString wxRichTextCompositeObject::GetTextForRange(const wxRichTextRange& range) const
336{
337 wxString text;
338 wxRichTextObjectList::compatibility_iterator node = m_children.GetFirst();
339 while (node)
340 {
341 wxRichTextObject* child = node->GetData();
342 wxRichTextRange childRange = range;
343 if (!child->GetRange().IsOutside(range))
344 {
345 childRange.LimitTo(child->GetRange());
7fe8059f 346
5d7836c4 347 wxString childText = child->GetTextForRange(childRange);
7fe8059f 348
5d7836c4
JS
349 text += childText;
350 }
351 node = node->GetNext();
352 }
353
354 return text;
355}
356
357/// Recursively merge all pieces that can be merged.
358bool wxRichTextCompositeObject::Defragment()
359{
360 wxRichTextObjectList::compatibility_iterator node = m_children.GetFirst();
361 while (node)
362 {
363 wxRichTextObject* child = node->GetData();
364 wxRichTextCompositeObject* composite = wxDynamicCast(child, wxRichTextCompositeObject);
7fe8059f 365 if (composite)
5d7836c4
JS
366 composite->Defragment();
367
368 if (node->GetNext())
369 {
370 wxRichTextObject* nextChild = node->GetNext()->GetData();
371 if (child->CanMerge(nextChild) && child->Merge(nextChild))
372 {
373 nextChild->Dereference();
9e31a660 374 m_children.Erase(node->GetNext());
5d7836c4
JS
375
376 // Don't set node -- we'll see if we can merge again with the next
377 // child.
378 }
379 else
380 node = node->GetNext();
381 }
382 else
383 node = node->GetNext();
384 }
385
386 return true;
387}
388
389/// Dump to output stream for debugging
390void wxRichTextCompositeObject::Dump(wxTextOutputStream& stream)
391{
392 wxRichTextObjectList::compatibility_iterator node = m_children.GetFirst();
393 while (node)
394 {
395 wxRichTextObject* child = node->GetData();
396 child->Dump(stream);
397 node = node->GetNext();
398 }
399}
400
401
402/*!
403 * wxRichTextBox
404 * This defines a 2D space to lay out objects
405 */
406
407IMPLEMENT_DYNAMIC_CLASS(wxRichTextBox, wxRichTextCompositeObject)
408
409wxRichTextBox::wxRichTextBox(wxRichTextObject* parent):
410 wxRichTextCompositeObject(parent)
411{
412}
413
414/// Draw the item
415bool wxRichTextBox::Draw(wxDC& dc, const wxRichTextRange& range, const wxRichTextRange& selectionRange, const wxRect& WXUNUSED(rect), int descent, int style)
416{
417 wxRichTextObjectList::compatibility_iterator node = m_children.GetFirst();
418 while (node)
419 {
420 wxRichTextObject* child = node->GetData();
421
422 wxRect childRect = wxRect(child->GetPosition(), child->GetCachedSize());
423 child->Draw(dc, range, selectionRange, childRect, descent, style);
424
425 node = node->GetNext();
426 }
427 return true;
428}
429
430/// Lay the item out
38113684 431bool wxRichTextBox::Layout(wxDC& dc, const wxRect& rect, int style)
5d7836c4
JS
432{
433 wxRichTextObjectList::compatibility_iterator node = m_children.GetFirst();
434 while (node)
435 {
436 wxRichTextObject* child = node->GetData();
38113684 437 child->Layout(dc, rect, style);
5d7836c4
JS
438
439 node = node->GetNext();
440 }
441 m_dirty = false;
442 return true;
443}
444
445/// Get/set the size for the given range. Assume only has one child.
7f0d9d71 446bool wxRichTextBox::GetRangeSize(const wxRichTextRange& range, wxSize& size, int& descent, wxDC& dc, int flags, wxPoint position) const
5d7836c4
JS
447{
448 wxRichTextObjectList::compatibility_iterator node = m_children.GetFirst();
449 if (node)
450 {
451 wxRichTextObject* child = node->GetData();
7f0d9d71 452 return child->GetRangeSize(range, size, descent, dc, flags, position);
5d7836c4
JS
453 }
454 else
455 return false;
456}
457
458/// Copy
459void wxRichTextBox::Copy(const wxRichTextBox& obj)
460{
461 wxRichTextCompositeObject::Copy(obj);
462}
463
464
465/*!
466 * wxRichTextParagraphLayoutBox
467 * This box knows how to lay out paragraphs.
468 */
469
470IMPLEMENT_DYNAMIC_CLASS(wxRichTextParagraphLayoutBox, wxRichTextBox)
471
472wxRichTextParagraphLayoutBox::wxRichTextParagraphLayoutBox(wxRichTextObject* parent):
473 wxRichTextBox(parent)
474{
475 Init();
476}
477
478/// Initialize the object.
479void wxRichTextParagraphLayoutBox::Init()
480{
481 m_ctrl = NULL;
482
483 // For now, assume is the only box and has no initial size.
484 m_range = wxRichTextRange(0, -1);
485
38113684 486 m_invalidRange.SetRange(-1, -1);
5d7836c4
JS
487 m_leftMargin = 4;
488 m_rightMargin = 4;
489 m_topMargin = 4;
490 m_bottomMargin = 4;
491}
492
493/// Draw the item
011b3dcb 494bool wxRichTextParagraphLayoutBox::Draw(wxDC& dc, const wxRichTextRange& range, const wxRichTextRange& selectionRange, const wxRect& rect, int descent, int style)
5d7836c4
JS
495{
496 wxRichTextObjectList::compatibility_iterator node = m_children.GetFirst();
497 while (node)
498 {
499 wxRichTextParagraph* child = wxDynamicCast(node->GetData(), wxRichTextParagraph);
500 wxASSERT (child != NULL);
7fe8059f 501
5d7836c4
JS
502 if (child && !child->GetRange().IsOutside(range))
503 {
504 wxRect childRect(child->GetPosition(), child->GetCachedSize());
7fe8059f 505
011b3dcb
JS
506 if (childRect.GetTop() > rect.GetBottom() || childRect.GetBottom() < rect.GetTop())
507 {
508 // Skip
509 }
510 else
511 child->Draw(dc, child->GetRange(), selectionRange, childRect, descent, style);
5d7836c4
JS
512 }
513
514 node = node->GetNext();
515 }
516 return true;
517}
518
519/// Lay the item out
38113684 520bool wxRichTextParagraphLayoutBox::Layout(wxDC& dc, const wxRect& rect, int style)
5d7836c4 521{
4d551ad5
JS
522 wxRect availableSpace;
523 bool formatRect = (style & wxRICHTEXT_LAYOUT_SPECIFIED_RECT) == wxRICHTEXT_LAYOUT_SPECIFIED_RECT;
524
525 // If only laying out a specific area, the passed rect has a different meaning:
526 // the visible part of the buffer.
527 if (formatRect)
528 {
529 availableSpace = wxRect(0 + m_leftMargin,
530 0 + m_topMargin,
531 rect.width - m_leftMargin - m_rightMargin,
532 rect.height);
533
534 // Invalidate the part of the buffer from the first visible line
535 // to the end. If other parts of the buffer are currently invalid,
536 // then they too will be taken into account if they are above
537 // the visible point.
538 long startPos = 0;
539 wxRichTextLine* line = GetLineAtYPosition(rect.y);
540 if (line)
541 startPos = line->GetAbsoluteRange().GetStart();
542
543 Invalidate(wxRichTextRange(startPos, GetRange().GetEnd()));
544 }
545 else
546 availableSpace = wxRect(rect.x + m_leftMargin,
5d7836c4
JS
547 rect.y + m_topMargin,
548 rect.width - m_leftMargin - m_rightMargin,
549 rect.height - m_topMargin - m_bottomMargin);
550
551 int maxWidth = 0;
7fe8059f 552
5d7836c4 553 wxRichTextObjectList::compatibility_iterator node = m_children.GetFirst();
39a1c2f2 554
38113684 555 bool layoutAll = true;
1e967276 556
38113684
JS
557 // Get invalid range, rounding to paragraph start/end.
558 wxRichTextRange invalidRange = GetInvalidRange(true);
559
4d551ad5 560 if (invalidRange == wxRICHTEXT_NONE && !formatRect)
1e967276
JS
561 return true;
562
563 if (invalidRange == wxRICHTEXT_ALL)
564 layoutAll = true;
38113684
JS
565 else // If we know what range is affected, start laying out from that point on.
566 if (invalidRange.GetStart() > GetRange().GetStart())
2c375f42 567 {
38113684 568 wxRichTextParagraph* firstParagraph = GetParagraphAtPosition(invalidRange.GetStart());
2c375f42
JS
569 if (firstParagraph)
570 {
571 wxRichTextObjectList::compatibility_iterator firstNode = m_children.Find(firstParagraph);
0cc70962
VZ
572 wxRichTextObjectList::compatibility_iterator previousNode;
573 if ( firstNode )
574 previousNode = firstNode->GetPrevious();
2c375f42
JS
575 if (firstNode && previousNode)
576 {
577 wxRichTextParagraph* previousParagraph = wxDynamicCast(previousNode->GetData(), wxRichTextParagraph);
578 availableSpace.y = previousParagraph->GetPosition().y + previousParagraph->GetCachedSize().y;
7fe8059f 579
2c375f42
JS
580 // Now we're going to start iterating from the first affected paragraph.
581 node = firstNode;
1e967276
JS
582
583 layoutAll = false;
2c375f42
JS
584 }
585 }
586 }
587
4d551ad5
JS
588 // A way to force speedy rest-of-buffer layout (the 'else' below)
589 bool forceQuickLayout = false;
39a1c2f2 590
5d7836c4
JS
591 while (node)
592 {
593 // Assume this box only contains paragraphs
594
595 wxRichTextParagraph* child = wxDynamicCast(node->GetData(), wxRichTextParagraph);
f120af9d 596 wxCHECK_MSG( child, false, _T("Unknown object in layout") );
7fe8059f 597
1e967276 598 // TODO: what if the child hasn't been laid out (e.g. involved in Undo) but still has 'old' lines
f120af9d
VZ
599 if ( !forceQuickLayout &&
600 (layoutAll ||
601 child->GetLines().IsEmpty() ||
602 !child->GetRange().IsOutside(invalidRange)) )
2c375f42 603 {
38113684 604 child->Layout(dc, availableSpace, style);
5d7836c4 605
2c375f42
JS
606 // Layout must set the cached size
607 availableSpace.y += child->GetCachedSize().y;
608 maxWidth = wxMax(maxWidth, child->GetCachedSize().x);
4d551ad5
JS
609
610 // If we're just formatting the visible part of the buffer,
611 // and we're now past the bottom of the window, start quick
612 // layout.
613 if (formatRect && child->GetPosition().y > rect.GetBottom())
614 forceQuickLayout = true;
2c375f42
JS
615 }
616 else
617 {
618 // We're outside the immediately affected range, so now let's just
619 // move everything up or down. This assumes that all the children have previously
620 // been laid out and have wrapped line lists associated with them.
621 // TODO: check all paragraphs before the affected range.
7fe8059f 622
2c375f42 623 int inc = availableSpace.y - child->GetPosition().y;
7fe8059f 624
2c375f42
JS
625 while (node)
626 {
627 wxRichTextParagraph* child = wxDynamicCast(node->GetData(), wxRichTextParagraph);
628 if (child)
629 {
630 if (child->GetLines().GetCount() == 0)
38113684 631 child->Layout(dc, availableSpace, style);
2c375f42
JS
632 else
633 child->SetPosition(wxPoint(child->GetPosition().x, child->GetPosition().y + inc));
7fe8059f 634
2c375f42
JS
635 availableSpace.y += child->GetCachedSize().y;
636 maxWidth = wxMax(maxWidth, child->GetCachedSize().x);
637 }
7fe8059f
WS
638
639 node = node->GetNext();
2c375f42
JS
640 }
641 break;
642 }
5d7836c4
JS
643
644 node = node->GetNext();
645 }
646
647 SetCachedSize(wxSize(maxWidth, availableSpace.y));
648
649 m_dirty = false;
1e967276 650 m_invalidRange = wxRICHTEXT_NONE;
5d7836c4
JS
651
652 return true;
653}
654
655/// Copy
656void wxRichTextParagraphLayoutBox::Copy(const wxRichTextParagraphLayoutBox& obj)
657{
658 wxRichTextBox::Copy(obj);
659}
660
661/// Get/set the size for the given range.
7f0d9d71 662bool wxRichTextParagraphLayoutBox::GetRangeSize(const wxRichTextRange& range, wxSize& size, int& descent, wxDC& dc, int flags, wxPoint position) const
5d7836c4
JS
663{
664 wxSize sz;
665
09f14108
JS
666 wxRichTextObjectList::compatibility_iterator startPara = wxRichTextObjectList::compatibility_iterator();
667 wxRichTextObjectList::compatibility_iterator endPara = wxRichTextObjectList::compatibility_iterator();
5d7836c4
JS
668
669 // First find the first paragraph whose starting position is within the range.
670 wxRichTextObjectList::compatibility_iterator node = m_children.GetFirst();
671 while (node)
672 {
673 // child is a paragraph
674 wxRichTextObject* child = node->GetData();
675 const wxRichTextRange& r = child->GetRange();
676
677 if (r.GetStart() <= range.GetStart() && r.GetEnd() >= range.GetStart())
678 {
679 startPara = node;
680 break;
681 }
682
683 node = node->GetNext();
684 }
685
686 // Next find the last paragraph containing part of the range
687 node = m_children.GetFirst();
688 while (node)
689 {
690 // child is a paragraph
691 wxRichTextObject* child = node->GetData();
692 const wxRichTextRange& r = child->GetRange();
693
694 if (r.GetStart() <= range.GetEnd() && r.GetEnd() >= range.GetEnd())
695 {
696 endPara = node;
697 break;
698 }
699
700 node = node->GetNext();
701 }
702
703 if (!startPara || !endPara)
704 return false;
705
706 // Now we can add up the sizes
707 for (node = startPara; node ; node = node->GetNext())
708 {
709 // child is a paragraph
710 wxRichTextObject* child = node->GetData();
711 const wxRichTextRange& childRange = child->GetRange();
712 wxRichTextRange rangeToFind = range;
713 rangeToFind.LimitTo(childRange);
714
715 wxSize childSize;
716
717 int childDescent = 0;
7f0d9d71 718 child->GetRangeSize(rangeToFind, childSize, childDescent, dc, flags, position);
5d7836c4
JS
719
720 descent = wxMax(childDescent, descent);
721
722 sz.x = wxMax(sz.x, childSize.x);
723 sz.y += childSize.y;
724
725 if (node == endPara)
726 break;
727 }
728
729 size = sz;
730
731 return true;
732}
733
734/// Get the paragraph at the given position
735wxRichTextParagraph* wxRichTextParagraphLayoutBox::GetParagraphAtPosition(long pos, bool caretPosition) const
736{
737 if (caretPosition)
738 pos ++;
739
740 // First find the first paragraph whose starting position is within the range.
741 wxRichTextObjectList::compatibility_iterator node = m_children.GetFirst();
742 while (node)
743 {
744 // child is a paragraph
745 wxRichTextParagraph* child = wxDynamicCast(node->GetData(), wxRichTextParagraph);
746 wxASSERT (child != NULL);
747
748 // Return first child in buffer if position is -1
749 // if (pos == -1)
750 // return child;
751
752 if (child->GetRange().Contains(pos))
753 return child;
754
755 node = node->GetNext();
756 }
757 return NULL;
758}
759
760/// Get the line at the given position
761wxRichTextLine* wxRichTextParagraphLayoutBox::GetLineAtPosition(long pos, bool caretPosition) const
762{
763 if (caretPosition)
764 pos ++;
765
766 // First find the first paragraph whose starting position is within the range.
767 wxRichTextObjectList::compatibility_iterator node = m_children.GetFirst();
768 while (node)
769 {
770 // child is a paragraph
771 wxRichTextParagraph* child = wxDynamicCast(node->GetData(), wxRichTextParagraph);
772 wxASSERT (child != NULL);
773
774 wxRichTextLineList::compatibility_iterator node2 = child->GetLines().GetFirst();
775 while (node2)
776 {
777 wxRichTextLine* line = node2->GetData();
778
1e967276
JS
779 wxRichTextRange range = line->GetAbsoluteRange();
780
781 if (range.Contains(pos) ||
5d7836c4
JS
782
783 // If the position is end-of-paragraph, then return the last line of
784 // of the paragraph.
1e967276 785 (range.GetEnd() == child->GetRange().GetEnd()-1) && (pos == child->GetRange().GetEnd()))
5d7836c4
JS
786 return line;
787
788 node2 = node2->GetNext();
7fe8059f 789 }
5d7836c4
JS
790
791 node = node->GetNext();
792 }
793
794 int lineCount = GetLineCount();
795 if (lineCount > 0)
796 return GetLineForVisibleLineNumber(lineCount-1);
797 else
798 return NULL;
799}
800
801/// Get the line at the given y pixel position, or the last line.
802wxRichTextLine* wxRichTextParagraphLayoutBox::GetLineAtYPosition(int y) const
803{
804 wxRichTextObjectList::compatibility_iterator node = m_children.GetFirst();
805 while (node)
806 {
807 wxRichTextParagraph* child = wxDynamicCast(node->GetData(), wxRichTextParagraph);
808 wxASSERT (child != NULL);
809
810 wxRichTextLineList::compatibility_iterator node2 = child->GetLines().GetFirst();
811 while (node2)
812 {
813 wxRichTextLine* line = node2->GetData();
814
815 wxRect rect(line->GetRect());
816
817 if (y <= rect.GetBottom())
818 return line;
819
820 node2 = node2->GetNext();
7fe8059f 821 }
5d7836c4
JS
822
823 node = node->GetNext();
824 }
825
826 // Return last line
827 int lineCount = GetLineCount();
828 if (lineCount > 0)
829 return GetLineForVisibleLineNumber(lineCount-1);
830 else
831 return NULL;
832}
833
834/// Get the number of visible lines
835int wxRichTextParagraphLayoutBox::GetLineCount() const
836{
837 int count = 0;
838
839 wxRichTextObjectList::compatibility_iterator node = m_children.GetFirst();
840 while (node)
841 {
842 wxRichTextParagraph* child = wxDynamicCast(node->GetData(), wxRichTextParagraph);
843 wxASSERT (child != NULL);
844
845 count += child->GetLines().GetCount();
846 node = node->GetNext();
847 }
848 return count;
849}
850
851
852/// Get the paragraph for a given line
853wxRichTextParagraph* wxRichTextParagraphLayoutBox::GetParagraphForLine(wxRichTextLine* line) const
854{
1e967276 855 return GetParagraphAtPosition(line->GetAbsoluteRange().GetStart());
5d7836c4
JS
856}
857
858/// Get the line size at the given position
859wxSize wxRichTextParagraphLayoutBox::GetLineSizeAtPosition(long pos, bool caretPosition) const
860{
861 wxRichTextLine* line = GetLineAtPosition(pos, caretPosition);
862 if (line)
863 {
864 return line->GetSize();
865 }
866 else
867 return wxSize(0, 0);
868}
869
870
871/// Convenience function to add a paragraph of text
872wxRichTextRange wxRichTextParagraphLayoutBox::AddParagraph(const wxString& text)
873{
874 wxTextAttrEx style(GetAttributes());
875
876 // Apply default style. If the style has no attributes set,
877 // then the attributes will remain the 'basic style' (i.e. the
878 // layout box's style).
879 wxRichTextApplyStyle(style, GetDefaultStyle());
880
881 wxRichTextParagraph* para = new wxRichTextParagraph(text, this, & style);
882
883 AppendChild(para);
884
885 UpdateRanges();
886 SetDirty(true);
887
888 return para->GetRange();
889}
890
891/// Adds multiple paragraphs, based on newlines.
892wxRichTextRange wxRichTextParagraphLayoutBox::AddParagraphs(const wxString& text)
893{
894 wxTextAttrEx style(GetAttributes());
895 //wxLogDebug("Initial style = %s", style.GetFont().GetFaceName());
896 //wxLogDebug("Initial size = %d", style.GetFont().GetPointSize());
897
898 // Apply default style. If the style has no attributes set,
899 // then the attributes will remain the 'basic style' (i.e. the
900 // layout box's style).
901 wxRichTextApplyStyle(style, GetDefaultStyle());
902
903 //wxLogDebug("Style after applying default style = %s", style.GetFont().GetFaceName());
904 //wxLogDebug("Size after applying default style = %d", style.GetFont().GetPointSize());
905
906 wxRichTextParagraph* firstPara = NULL;
907 wxRichTextParagraph* lastPara = NULL;
908
909 wxRichTextRange range(-1, -1);
910 size_t i = 0;
28f92d74 911 size_t len = text.length();
5d7836c4
JS
912 wxString line;
913 while (i < len)
914 {
915 wxChar ch = text[i];
916 if (ch == wxT('\n') || ch == wxT('\r'))
917 {
918 wxRichTextParagraph* para = new wxRichTextParagraph(line, this, & style);
919
920 AppendChild(para);
921 if (!firstPara)
922 firstPara = para;
923 lastPara = para;
924 line = wxEmptyString;
925 }
926 else
927 line += ch;
928
929 i ++;
930 }
7fe8059f 931 if (!line.empty())
5d7836c4
JS
932 {
933 lastPara = new wxRichTextParagraph(line, this, & style);
934 //wxLogDebug("Para Face = %s", lastPara->GetAttributes().GetFont().GetFaceName());
935 AppendChild(lastPara);
936 }
937
938 if (firstPara)
939 range.SetStart(firstPara->GetRange().GetStart());
940 else if (lastPara)
941 range.SetStart(lastPara->GetRange().GetStart());
942
943 if (lastPara)
944 range.SetEnd(lastPara->GetRange().GetEnd());
945 else if (firstPara)
946 range.SetEnd(firstPara->GetRange().GetEnd());
947
948 UpdateRanges();
949 SetDirty(false);
950
951 return GetRange();
952}
953
954/// Convenience function to add an image
955wxRichTextRange wxRichTextParagraphLayoutBox::AddImage(const wxImage& image)
956{
957 wxTextAttrEx style(GetAttributes());
958
959 // Apply default style. If the style has no attributes set,
960 // then the attributes will remain the 'basic style' (i.e. the
961 // layout box's style).
962 wxRichTextApplyStyle(style, GetDefaultStyle());
963
964 wxRichTextParagraph* para = new wxRichTextParagraph(this, & style);
965 AppendChild(para);
966 para->AppendChild(new wxRichTextImage(image, this));
967
968 UpdateRanges();
969 SetDirty(true);
970
971 return para->GetRange();
972}
973
974
975/// Insert fragment into this box at the given position. If partialParagraph is true,
976/// it is assumed that the last (or only) paragraph is just a piece of data with no paragraph
977/// marker.
978/// TODO: if fragment is inserted inside styled fragment, must apply that style to
979/// to the data (if it has a default style, anyway).
980
981bool wxRichTextParagraphLayoutBox::InsertFragment(long position, wxRichTextFragment& fragment)
982{
983 SetDirty(true);
984
985 // First, find the first paragraph whose starting position is within the range.
986 wxRichTextParagraph* para = GetParagraphAtPosition(position);
987 if (para)
988 {
989 wxRichTextObjectList::compatibility_iterator node = m_children.Find(para);
990
991 // Now split at this position, returning the object to insert the new
992 // ones in front of.
993 wxRichTextObject* nextObject = para->SplitAt(position);
994
995 // Special case: partial paragraph, just one paragraph. Might be a small amount of
996 // text, for example, so let's optimize.
997
998 if (fragment.GetPartialParagraph() && fragment.GetChildren().GetCount() == 1)
999 {
1000 // Add the first para to this para...
1001 wxRichTextObjectList::compatibility_iterator firstParaNode = fragment.GetChildren().GetFirst();
1002 if (!firstParaNode)
1003 return false;
1004
1005 // Iterate through the fragment paragraph inserting the content into this paragraph.
1006 wxRichTextParagraph* firstPara = wxDynamicCast(firstParaNode->GetData(), wxRichTextParagraph);
1007 wxASSERT (firstPara != NULL);
1008
1009 wxRichTextObjectList::compatibility_iterator objectNode = firstPara->GetChildren().GetFirst();
1010 while (objectNode)
1011 {
1012 wxRichTextObject* newObj = objectNode->GetData()->Clone();
7fe8059f 1013
5d7836c4
JS
1014 if (!nextObject)
1015 {
1016 // Append
1017 para->AppendChild(newObj);
1018 }
1019 else
1020 {
1021 // Insert before nextObject
1022 para->InsertChild(newObj, nextObject);
1023 }
7fe8059f 1024
5d7836c4
JS
1025 objectNode = objectNode->GetNext();
1026 }
1027
1028 return true;
1029 }
1030 else
1031 {
1032 // Procedure for inserting a fragment consisting of a number of
1033 // paragraphs:
1034 //
1035 // 1. Remove and save the content that's after the insertion point, for adding
1036 // back once we've added the fragment.
1037 // 2. Add the content from the first fragment paragraph to the current
1038 // paragraph.
1039 // 3. Add remaining fragment paragraphs after the current paragraph.
1040 // 4. Add back the saved content from the first paragraph. If partialParagraph
1041 // is true, add it to the last paragraph added and not a new one.
1042
1043 // 1. Remove and save objects after split point.
1044 wxList savedObjects;
1045 if (nextObject)
1046 para->MoveToList(nextObject, savedObjects);
1047
1048 // 2. Add the content from the 1st fragment paragraph.
1049 wxRichTextObjectList::compatibility_iterator firstParaNode = fragment.GetChildren().GetFirst();
1050 if (!firstParaNode)
1051 return false;
1052
1053 wxRichTextParagraph* firstPara = wxDynamicCast(firstParaNode->GetData(), wxRichTextParagraph);
1054 wxASSERT(firstPara != NULL);
1055
1056 wxRichTextObjectList::compatibility_iterator objectNode = firstPara->GetChildren().GetFirst();
1057 while (objectNode)
1058 {
1059 wxRichTextObject* newObj = objectNode->GetData()->Clone();
7fe8059f 1060
5d7836c4
JS
1061 // Append
1062 para->AppendChild(newObj);
7fe8059f 1063
5d7836c4
JS
1064 objectNode = objectNode->GetNext();
1065 }
1066
1067 // 3. Add remaining fragment paragraphs after the current paragraph.
1068 wxRichTextObjectList::compatibility_iterator nextParagraphNode = node->GetNext();
1069 wxRichTextObject* nextParagraph = NULL;
1070 if (nextParagraphNode)
1071 nextParagraph = nextParagraphNode->GetData();
1072
1073 wxRichTextObjectList::compatibility_iterator i = fragment.GetChildren().GetFirst()->GetNext();
1074 wxRichTextParagraph* finalPara = para;
1075
1076 // If there was only one paragraph, we need to insert a new one.
1077 if (!i)
1078 {
1079 finalPara = new wxRichTextParagraph;
1080
1081 // TODO: These attributes should come from the subsequent paragraph
1082 // when originally deleted, since the subsequent para takes on
1083 // the previous para's attributes.
1084 finalPara->SetAttributes(firstPara->GetAttributes());
1085
1086 if (nextParagraph)
1087 InsertChild(finalPara, nextParagraph);
1088 else
7fe8059f 1089 AppendChild(finalPara);
5d7836c4
JS
1090 }
1091 else while (i)
1092 {
1093 wxRichTextParagraph* para = wxDynamicCast(i->GetData(), wxRichTextParagraph);
1094 wxASSERT( para != NULL );
1095
1096 finalPara = (wxRichTextParagraph*) para->Clone();
1097
1098 if (nextParagraph)
1099 InsertChild(finalPara, nextParagraph);
1100 else
1101 AppendChild(finalPara);
7fe8059f 1102
5d7836c4
JS
1103 i = i->GetNext();
1104 }
1105
1106 // 4. Add back the remaining content.
1107 if (finalPara)
1108 {
1109 finalPara->MoveFromList(savedObjects);
1110
1111 // Ensure there's at least one object
1112 if (finalPara->GetChildCount() == 0)
1113 {
7fe8059f 1114 wxRichTextPlainText* text = new wxRichTextPlainText(wxEmptyString);
5d7836c4
JS
1115 text->SetAttributes(finalPara->GetAttributes());
1116
1117 finalPara->AppendChild(text);
1118 }
1119 }
1120
1121 return true;
1122 }
1123 }
1124 else
1125 {
1126 // Append
1127 wxRichTextObjectList::compatibility_iterator i = fragment.GetChildren().GetFirst();
1128 while (i)
1129 {
1130 wxRichTextParagraph* para = wxDynamicCast(i->GetData(), wxRichTextParagraph);
1131 wxASSERT( para != NULL );
7fe8059f 1132
5d7836c4 1133 AppendChild(para->Clone());
7fe8059f 1134
5d7836c4
JS
1135 i = i->GetNext();
1136 }
1137
1138 return true;
1139 }
5d7836c4
JS
1140}
1141
1142/// Make a copy of the fragment corresponding to the given range, putting it in 'fragment'.
1143/// If there was an incomplete paragraph at the end, partialParagraph is set to true.
1144bool wxRichTextParagraphLayoutBox::CopyFragment(const wxRichTextRange& range, wxRichTextFragment& fragment)
1145{
1146 wxRichTextObjectList::compatibility_iterator i = GetChildren().GetFirst();
1147 while (i)
1148 {
1149 wxRichTextParagraph* para = wxDynamicCast(i->GetData(), wxRichTextParagraph);
1150 wxASSERT( para != NULL );
1151
1152 if (!para->GetRange().IsOutside(range))
1153 {
1154 fragment.AppendChild(para->Clone());
7fe8059f 1155 }
5d7836c4
JS
1156 i = i->GetNext();
1157 }
1158
1159 // Now top and tail the first and last paragraphs in our new fragment (which might be the same).
1160 if (!fragment.IsEmpty())
1161 {
1162 wxRichTextRange topTailRange(range);
1163
1164 wxRichTextParagraph* firstPara = wxDynamicCast(fragment.GetChildren().GetFirst()->GetData(), wxRichTextParagraph);
1165 wxASSERT( firstPara != NULL );
1166
1167 // Chop off the start of the paragraph
1168 if (topTailRange.GetStart() > firstPara->GetRange().GetStart())
1169 {
1170 wxRichTextRange r(firstPara->GetRange().GetStart(), topTailRange.GetStart()-1);
1171 firstPara->DeleteRange(r);
1172
1173 // Make sure the numbering is correct
1174 long end;
1175 fragment.CalculateRange(firstPara->GetRange().GetStart(), end);
1176
1177 // Now, we've deleted some positions, so adjust the range
1178 // accordingly.
1179 topTailRange.SetEnd(topTailRange.GetEnd() - r.GetLength());
1180 }
1181
1182 wxRichTextParagraph* lastPara = wxDynamicCast(fragment.GetChildren().GetLast()->GetData(), wxRichTextParagraph);
1183 wxASSERT( lastPara != NULL );
1184
1185 if (topTailRange.GetEnd() < (lastPara->GetRange().GetEnd()-1))
1186 {
1187 wxRichTextRange r(topTailRange.GetEnd()+1, lastPara->GetRange().GetEnd()-1); /* -1 since actual text ends 1 position before end of para marker */
1188 lastPara->DeleteRange(r);
1189
1190 // Make sure the numbering is correct
1191 long end;
1192 fragment.CalculateRange(firstPara->GetRange().GetStart(), end);
1193
1194 // We only have part of a paragraph at the end
1195 fragment.SetPartialParagraph(true);
1196 }
1197 else
1198 {
1199 if (topTailRange.GetEnd() == (lastPara->GetRange().GetEnd() - 1))
1200 // We have a partial paragraph (don't save last new paragraph marker)
1201 fragment.SetPartialParagraph(true);
1202 else
1203 // We have a complete paragraph
1204 fragment.SetPartialParagraph(false);
1205 }
1206 }
1207
1208 return true;
1209}
1210
1211/// Given a position, get the number of the visible line (potentially many to a paragraph),
1212/// starting from zero at the start of the buffer.
1213long wxRichTextParagraphLayoutBox::GetVisibleLineNumber(long pos, bool caretPosition, bool startOfLine) const
1214{
1215 if (caretPosition)
1216 pos ++;
1217
1218 int lineCount = 0;
1219
1220 wxRichTextObjectList::compatibility_iterator node = m_children.GetFirst();
1221 while (node)
1222 {
1223 wxRichTextParagraph* child = wxDynamicCast(node->GetData(), wxRichTextParagraph);
1224 wxASSERT( child != NULL );
1225
1226 if (child->GetRange().Contains(pos))
1227 {
1228 wxRichTextLineList::compatibility_iterator node2 = child->GetLines().GetFirst();
1229 while (node2)
1230 {
1231 wxRichTextLine* line = node2->GetData();
1e967276 1232 wxRichTextRange lineRange = line->GetAbsoluteRange();
7fe8059f 1233
1e967276 1234 if (lineRange.Contains(pos))
5d7836c4
JS
1235 {
1236 // If the caret is displayed at the end of the previous wrapped line,
1237 // we want to return the line it's _displayed_ at (not the actual line
1238 // containing the position).
1e967276 1239 if (lineRange.GetStart() == pos && !startOfLine && child->GetRange().GetStart() != pos)
5d7836c4
JS
1240 return lineCount - 1;
1241 else
1242 return lineCount;
1243 }
1244
1245 lineCount ++;
7fe8059f 1246
5d7836c4
JS
1247 node2 = node2->GetNext();
1248 }
1249 // If we didn't find it in the lines, it must be
1250 // the last position of the paragraph. So return the last line.
1251 return lineCount-1;
1252 }
1253 else
1254 lineCount += child->GetLines().GetCount();
1255
1256 node = node->GetNext();
1257 }
1258
1259 // Not found
1260 return -1;
1261}
1262
1263/// Given a line number, get the corresponding wxRichTextLine object.
1264wxRichTextLine* wxRichTextParagraphLayoutBox::GetLineForVisibleLineNumber(long lineNumber) const
1265{
1266 int lineCount = 0;
1267
1268 wxRichTextObjectList::compatibility_iterator node = m_children.GetFirst();
1269 while (node)
1270 {
1271 wxRichTextParagraph* child = wxDynamicCast(node->GetData(), wxRichTextParagraph);
1272 wxASSERT(child != NULL);
1273
1274 if (lineNumber < (int) (child->GetLines().GetCount() + lineCount))
1275 {
1276 wxRichTextLineList::compatibility_iterator node2 = child->GetLines().GetFirst();
1277 while (node2)
1278 {
1279 wxRichTextLine* line = node2->GetData();
7fe8059f 1280
5d7836c4
JS
1281 if (lineCount == lineNumber)
1282 return line;
1283
1284 lineCount ++;
7fe8059f 1285
5d7836c4 1286 node2 = node2->GetNext();
7fe8059f 1287 }
5d7836c4
JS
1288 }
1289 else
1290 lineCount += child->GetLines().GetCount();
1291
1292 node = node->GetNext();
1293 }
1294
1295 // Didn't find it
1296 return NULL;
1297}
1298
1299/// Delete range from layout.
1300bool wxRichTextParagraphLayoutBox::DeleteRange(const wxRichTextRange& range)
1301{
1302 wxRichTextObjectList::compatibility_iterator node = m_children.GetFirst();
7fe8059f 1303
5d7836c4
JS
1304 while (node)
1305 {
1306 wxRichTextParagraph* obj = wxDynamicCast(node->GetData(), wxRichTextParagraph);
1307 wxASSERT (obj != NULL);
1308
1309 wxRichTextObjectList::compatibility_iterator next = node->GetNext();
7fe8059f 1310
5d7836c4
JS
1311 // Delete the range in each paragraph
1312
1313 if (!obj->GetRange().IsOutside(range))
1314 {
1315 // Deletes the content of this object within the given range
1316 obj->DeleteRange(range);
1317
1318 // If the whole paragraph is within the range to delete,
1319 // delete the whole thing.
1320 if (range.GetStart() <= obj->GetRange().GetStart() && range.GetEnd() >= obj->GetRange().GetEnd())
1321 {
1322 // Delete the whole object
1323 RemoveChild(obj, true);
1324 }
1325 // If the range includes the paragraph end, we need to join this
1326 // and the next paragraph.
1327 else if (range.Contains(obj->GetRange().GetEnd()))
1328 {
1329 // We need to move the objects from the next paragraph
1330 // to this paragraph
1331
1332 if (next)
1333 {
1334 wxRichTextParagraph* nextParagraph = wxDynamicCast(next->GetData(), wxRichTextParagraph);
1335 next = next->GetNext();
1336 if (nextParagraph)
1337 {
1338 // Delete the stuff we need to delete
1339 nextParagraph->DeleteRange(range);
1340
1341 // Move the objects to the previous para
1342 wxRichTextObjectList::compatibility_iterator node1 = nextParagraph->GetChildren().GetFirst();
1343
1344 while (node1)
1345 {
1346 wxRichTextObject* obj1 = node1->GetData();
1347
1348 // If the object is empty, optimise it out
1349 if (obj1->IsEmpty())
1350 {
1351 delete obj1;
1352 }
1353 else
1354 {
1355 obj->AppendChild(obj1);
1356 }
1357
1358 wxRichTextObjectList::compatibility_iterator next1 = node1->GetNext();
9e31a660 1359 nextParagraph->GetChildren().Erase(node1);
5d7836c4
JS
1360
1361 node1 = next1;
1362 }
1363
1364 // Delete the paragraph
1365 RemoveChild(nextParagraph, true);
1366
1367 }
7fe8059f 1368 }
5d7836c4
JS
1369
1370 }
1371 }
7fe8059f 1372
5d7836c4
JS
1373 node = next;
1374 }
7fe8059f 1375
5d7836c4
JS
1376 return true;
1377}
1378
1379/// Get any text in this object for the given range
1380wxString wxRichTextParagraphLayoutBox::GetTextForRange(const wxRichTextRange& range) const
1381{
1382 int lineCount = 0;
1383 wxString text;
1384 wxRichTextObjectList::compatibility_iterator node = m_children.GetFirst();
1385 while (node)
1386 {
1387 wxRichTextObject* child = node->GetData();
1388 if (!child->GetRange().IsOutside(range))
1389 {
1390 if (lineCount > 0)
1391 text += wxT("\n");
1392 wxRichTextRange childRange = range;
1393 childRange.LimitTo(child->GetRange());
7fe8059f 1394
5d7836c4 1395 wxString childText = child->GetTextForRange(childRange);
7fe8059f 1396
5d7836c4
JS
1397 text += childText;
1398
1399 lineCount ++;
1400 }
1401 node = node->GetNext();
1402 }
1403
1404 return text;
1405}
1406
1407/// Get all the text
1408wxString wxRichTextParagraphLayoutBox::GetText() const
1409{
1410 return GetTextForRange(GetRange());
1411}
1412
1413/// Get the paragraph by number
1414wxRichTextParagraph* wxRichTextParagraphLayoutBox::GetParagraphAtLine(long paragraphNumber) const
1415{
27e20452 1416 if ((size_t) paragraphNumber >= GetChildCount())
5d7836c4
JS
1417 return NULL;
1418
1419 return (wxRichTextParagraph*) GetChild((size_t) paragraphNumber);
1420}
1421
1422/// Get the length of the paragraph
1423int wxRichTextParagraphLayoutBox::GetParagraphLength(long paragraphNumber) const
1424{
1425 wxRichTextParagraph* para = GetParagraphAtLine(paragraphNumber);
1426 if (para)
1427 return para->GetRange().GetLength() - 1; // don't include newline
1428 else
1429 return 0;
1430}
1431
1432/// Get the text of the paragraph
1433wxString wxRichTextParagraphLayoutBox::GetParagraphText(long paragraphNumber) const
1434{
1435 wxRichTextParagraph* para = GetParagraphAtLine(paragraphNumber);
1436 if (para)
1437 return para->GetTextForRange(para->GetRange());
1438 else
1439 return wxEmptyString;
1440}
1441
1442/// Convert zero-based line column and paragraph number to a position.
1443long wxRichTextParagraphLayoutBox::XYToPosition(long x, long y) const
1444{
1445 wxRichTextParagraph* para = GetParagraphAtLine(y);
1446 if (para)
1447 {
1448 return para->GetRange().GetStart() + x;
1449 }
1450 else
1451 return -1;
1452}
1453
1454/// Convert zero-based position to line column and paragraph number
1455bool wxRichTextParagraphLayoutBox::PositionToXY(long pos, long* x, long* y) const
1456{
1457 wxRichTextParagraph* para = GetParagraphAtPosition(pos);
1458 if (para)
1459 {
1460 int count = 0;
1461 wxRichTextObjectList::compatibility_iterator node = m_children.GetFirst();
1462 while (node)
1463 {
1464 wxRichTextObject* child = node->GetData();
1465 if (child == para)
1466 break;
1467 count ++;
1468 node = node->GetNext();
1469 }
1470
1471 *y = count;
1472 *x = pos - para->GetRange().GetStart();
1473
1474 return true;
1475 }
1476 else
1477 return false;
1478}
1479
1480/// Get the leaf object in a paragraph at this position.
1481/// Given a line number, get the corresponding wxRichTextLine object.
1482wxRichTextObject* wxRichTextParagraphLayoutBox::GetLeafObjectAtPosition(long position) const
1483{
1484 wxRichTextParagraph* para = GetParagraphAtPosition(position);
1485 if (para)
1486 {
1487 wxRichTextObjectList::compatibility_iterator node = para->GetChildren().GetFirst();
7fe8059f 1488
5d7836c4
JS
1489 while (node)
1490 {
1491 wxRichTextObject* child = node->GetData();
1492 if (child->GetRange().Contains(position))
1493 return child;
7fe8059f 1494
5d7836c4
JS
1495 node = node->GetNext();
1496 }
1497 if (position == para->GetRange().GetEnd() && para->GetChildCount() > 0)
1498 return para->GetChildren().GetLast()->GetData();
1499 }
1500 return NULL;
1501}
1502
1503/// Set character or paragraph text attributes: apply character styles only to immediate text nodes
1504bool wxRichTextParagraphLayoutBox::SetStyle(const wxRichTextRange& range, const wxRichTextAttr& style, bool withUndo)
1505{
1506 bool characterStyle = false;
1507 bool paragraphStyle = false;
1508
1509 if (style.IsCharacterStyle())
1510 characterStyle = true;
1511 if (style.IsParagraphStyle())
1512 paragraphStyle = true;
1513
1514 // If we are associated with a control, make undoable; otherwise, apply immediately
1515 // to the data.
1516
1517 bool haveControl = (GetRichTextCtrl() != NULL);
1518
1519 wxRichTextAction* action = NULL;
7fe8059f 1520
5d7836c4
JS
1521 if (haveControl && withUndo)
1522 {
1523 action = new wxRichTextAction(NULL, _("Change Style"), wxRICHTEXT_CHANGE_STYLE, & GetRichTextCtrl()->GetBuffer(), GetRichTextCtrl());
1524 action->SetRange(range);
1525 action->SetPosition(GetRichTextCtrl()->GetCaretPosition());
1526 }
1527
1528 wxRichTextObjectList::compatibility_iterator node = m_children.GetFirst();
1529 while (node)
1530 {
1531 wxRichTextParagraph* para = wxDynamicCast(node->GetData(), wxRichTextParagraph);
1532 wxASSERT (para != NULL);
1533
1534 if (para && para->GetChildCount() > 0)
1535 {
1536 // Stop searching if we're beyond the range of interest
1537 if (para->GetRange().GetStart() > range.GetEnd())
1538 break;
1539
1540 if (!para->GetRange().IsOutside(range))
1541 {
1542 // We'll be using a copy of the paragraph to make style changes,
1543 // not updating the buffer directly.
4e09ebe8 1544 wxRichTextParagraph* newPara wxDUMMY_INITIALIZE(NULL);
7fe8059f 1545
5d7836c4
JS
1546 if (haveControl && withUndo)
1547 {
1548 newPara = new wxRichTextParagraph(*para);
1549 action->GetNewParagraphs().AppendChild(newPara);
1550
1551 // Also store the old ones for Undo
1552 action->GetOldParagraphs().AppendChild(new wxRichTextParagraph(*para));
1553 }
1554 else
1555 newPara = para;
1556
1557 if (paragraphStyle)
1558 wxRichTextApplyStyle(newPara->GetAttributes(), style);
1559
1560 if (characterStyle && range.GetStart() != newPara->GetRange().GetEnd())
1561 {
1562 wxRichTextRange childRange(range);
1563 childRange.LimitTo(newPara->GetRange());
7fe8059f 1564
5d7836c4
JS
1565 // Find the starting position and if necessary split it so
1566 // we can start applying a different style.
1567 // TODO: check that the style actually changes or is different
1568 // from style outside of range
4e09ebe8
JS
1569 wxRichTextObject* firstObject wxDUMMY_INITIALIZE(NULL);
1570 wxRichTextObject* lastObject wxDUMMY_INITIALIZE(NULL);
7fe8059f 1571
5d7836c4
JS
1572 if (childRange.GetStart() == newPara->GetRange().GetStart())
1573 firstObject = newPara->GetChildren().GetFirst()->GetData();
1574 else
1575 firstObject = newPara->SplitAt(range.GetStart());
7fe8059f 1576
5d7836c4
JS
1577 // Increment by 1 because we're apply the style one _after_ the split point
1578 long splitPoint = childRange.GetEnd();
1579 if (splitPoint != newPara->GetRange().GetEnd())
1580 splitPoint ++;
7fe8059f 1581
5d7836c4
JS
1582 // Find last object
1583 if (splitPoint == newPara->GetRange().GetEnd() || splitPoint == (newPara->GetRange().GetEnd() - 1))
1584 lastObject = newPara->GetChildren().GetLast()->GetData();
1585 else
1586 // lastObject is set as a side-effect of splitting. It's
1587 // returned as the object before the new object.
1588 (void) newPara->SplitAt(splitPoint, & lastObject);
7fe8059f 1589
5d7836c4
JS
1590 wxASSERT(firstObject != NULL);
1591 wxASSERT(lastObject != NULL);
7fe8059f 1592
5d7836c4
JS
1593 if (!firstObject || !lastObject)
1594 continue;
7fe8059f 1595
5d7836c4
JS
1596 wxRichTextObjectList::compatibility_iterator firstNode = newPara->GetChildren().Find(firstObject);
1597 wxRichTextObjectList::compatibility_iterator lastNode = newPara->GetChildren().Find(lastObject);
7fe8059f 1598
4c9847e1
MW
1599 wxASSERT(firstNode);
1600 wxASSERT(lastNode);
7fe8059f 1601
5d7836c4 1602 wxRichTextObjectList::compatibility_iterator node2 = firstNode;
7fe8059f 1603
5d7836c4
JS
1604 while (node2)
1605 {
1606 wxRichTextObject* child = node2->GetData();
7fe8059f 1607
5d7836c4
JS
1608 wxRichTextApplyStyle(child->GetAttributes(), style);
1609 if (node2 == lastNode)
1610 break;
7fe8059f 1611
5d7836c4
JS
1612 node2 = node2->GetNext();
1613 }
1614 }
1615 }
1616 }
1617
1618 node = node->GetNext();
1619 }
1620
1621 // Do action, or delay it until end of batch.
1622 if (haveControl && withUndo)
1623 GetRichTextCtrl()->GetBuffer().SubmitAction(action);
1624
1625 return true;
1626}
1627
1628/// Set text attributes
1629bool wxRichTextParagraphLayoutBox::SetStyle(const wxRichTextRange& range, const wxTextAttrEx& style, bool withUndo)
1630{
1631 wxRichTextAttr richStyle = style;
1632 return SetStyle(range, richStyle, withUndo);
1633}
1634
1635/// Get the text attributes for this position.
1636bool wxRichTextParagraphLayoutBox::GetStyle(long position, wxTextAttrEx& style) const
1637{
4e09ebe8 1638 wxRichTextObject* obj wxDUMMY_INITIALIZE(NULL);
e191ee87 1639
5d7836c4
JS
1640 if (style.IsParagraphStyle())
1641 obj = GetParagraphAtPosition(position);
1642 else
1643 obj = GetLeafObjectAtPosition(position);
e191ee87 1644
5d7836c4
JS
1645 if (obj)
1646 {
1647 style = obj->GetAttributes();
1648 return true;
1649 }
1650 else
1651 return false;
1652}
1653
1654/// Get the text attributes for this position.
1655bool wxRichTextParagraphLayoutBox::GetStyle(long position, wxRichTextAttr& style) const
1656{
4e09ebe8 1657 wxRichTextObject* obj wxDUMMY_INITIALIZE(NULL);
e191ee87 1658
5d7836c4
JS
1659 if (style.IsParagraphStyle())
1660 obj = GetParagraphAtPosition(position);
1661 else
1662 obj = GetLeafObjectAtPosition(position);
e191ee87 1663
5d7836c4
JS
1664 if (obj)
1665 {
1666 style = obj->GetAttributes();
1667 return true;
1668 }
1669 else
1670 return false;
1671}
1672
1673/// Set default style
1674bool wxRichTextParagraphLayoutBox::SetDefaultStyle(const wxTextAttrEx& style)
1675{
05b4fddf
JS
1676 // keep the old attributes if the new style doesn't specify them unless the
1677 // new style is empty - then reset m_defaultStyle (as there is no other way
1678 // to do it)
1679 if ( style.IsDefault() )
1680 m_defaultAttributes = style;
1681 else
1682 m_defaultAttributes = wxTextAttrEx::CombineEx(style, m_defaultAttributes, NULL);
5d7836c4
JS
1683
1684 return true;
1685}
1686
1687/// Test if this whole range has character attributes of the specified kind. If any
1688/// of the attributes are different within the range, the test fails. You
1689/// can use this to implement, for example, bold button updating. style must have
1690/// flags indicating which attributes are of interest.
1691bool wxRichTextParagraphLayoutBox::HasCharacterAttributes(const wxRichTextRange& range, const wxRichTextAttr& style) const
1692{
1693 int foundCount = 0;
1694 int matchingCount = 0;
1695
1696 wxRichTextObjectList::compatibility_iterator node = m_children.GetFirst();
1697 while (node)
1698 {
1699 wxRichTextParagraph* para = wxDynamicCast(node->GetData(), wxRichTextParagraph);
1700 wxASSERT (para != NULL);
1701
1702 if (para)
1703 {
1704 // Stop searching if we're beyond the range of interest
1705 if (para->GetRange().GetStart() > range.GetEnd())
1706 return foundCount == matchingCount;
1707
1708 if (!para->GetRange().IsOutside(range))
1709 {
1710 wxRichTextObjectList::compatibility_iterator node2 = para->GetChildren().GetFirst();
1711
1712 while (node2)
1713 {
1714 wxRichTextObject* child = node2->GetData();
1715 if (!child->GetRange().IsOutside(range) && child->IsKindOf(CLASSINFO(wxRichTextPlainText)))
1716 {
1717 foundCount ++;
1718 if (wxTextAttrEqPartial(child->GetAttributes(), style, style.GetFlags()))
1719 matchingCount ++;
1720 }
1721
1722 node2 = node2->GetNext();
1723 }
1724 }
1725 }
1726
1727 node = node->GetNext();
1728 }
1729
1730 return foundCount == matchingCount;
1731}
1732
1733bool wxRichTextParagraphLayoutBox::HasCharacterAttributes(const wxRichTextRange& range, const wxTextAttrEx& style) const
1734{
1735 wxRichTextAttr richStyle = style;
1736 return HasCharacterAttributes(range, richStyle);
1737}
1738
1739/// Test if this whole range has paragraph attributes of the specified kind. If any
1740/// of the attributes are different within the range, the test fails. You
1741/// can use this to implement, for example, centering button updating. style must have
1742/// flags indicating which attributes are of interest.
1743bool wxRichTextParagraphLayoutBox::HasParagraphAttributes(const wxRichTextRange& range, const wxRichTextAttr& style) const
1744{
1745 int foundCount = 0;
1746 int matchingCount = 0;
1747
1748 wxRichTextObjectList::compatibility_iterator node = m_children.GetFirst();
1749 while (node)
1750 {
1751 wxRichTextParagraph* para = wxDynamicCast(node->GetData(), wxRichTextParagraph);
1752 wxASSERT (para != NULL);
1753
1754 if (para)
1755 {
1756 // Stop searching if we're beyond the range of interest
1757 if (para->GetRange().GetStart() > range.GetEnd())
1758 return foundCount == matchingCount;
1759
1760 if (!para->GetRange().IsOutside(range))
1761 {
1762 foundCount ++;
1763 if (wxTextAttrEqPartial(para->GetAttributes(), style, style.GetFlags()))
1764 matchingCount ++;
1765 }
1766 }
1767
1768 node = node->GetNext();
1769 }
1770 return foundCount == matchingCount;
1771}
1772
1773bool wxRichTextParagraphLayoutBox::HasParagraphAttributes(const wxRichTextRange& range, const wxTextAttrEx& style) const
1774{
1775 wxRichTextAttr richStyle = style;
1776 return HasParagraphAttributes(range, richStyle);
1777}
1778
1779void wxRichTextParagraphLayoutBox::Clear()
1780{
1781 DeleteChildren();
1782}
1783
1784void wxRichTextParagraphLayoutBox::Reset()
1785{
1786 Clear();
1787
7fe8059f 1788 AddParagraph(wxEmptyString);
5d7836c4
JS
1789}
1790
38113684
JS
1791/// Invalidate the buffer. With no argument, invalidates whole buffer.
1792void wxRichTextParagraphLayoutBox::Invalidate(const wxRichTextRange& invalidRange)
1793{
1794 SetDirty(true);
39a1c2f2 1795
1e967276 1796 if (invalidRange == wxRICHTEXT_ALL)
38113684 1797 {
1e967276 1798 m_invalidRange = wxRICHTEXT_ALL;
38113684
JS
1799 return;
1800 }
1e967276
JS
1801
1802 // Already invalidating everything
1803 if (m_invalidRange == wxRICHTEXT_ALL)
1804 return;
39a1c2f2 1805
1e967276 1806 if ((invalidRange.GetStart() < m_invalidRange.GetStart()) || m_invalidRange.GetStart() == -1)
38113684
JS
1807 m_invalidRange.SetStart(invalidRange.GetStart());
1808 if (invalidRange.GetEnd() > m_invalidRange.GetEnd())
1809 m_invalidRange.SetEnd(invalidRange.GetEnd());
1810}
1811
1812/// Get invalid range, rounding to entire paragraphs if argument is true.
1813wxRichTextRange wxRichTextParagraphLayoutBox::GetInvalidRange(bool wholeParagraphs) const
1814{
1e967276 1815 if (m_invalidRange == wxRICHTEXT_ALL || m_invalidRange == wxRICHTEXT_NONE)
38113684 1816 return m_invalidRange;
39a1c2f2 1817
38113684 1818 wxRichTextRange range = m_invalidRange;
39a1c2f2 1819
38113684
JS
1820 if (wholeParagraphs)
1821 {
1822 wxRichTextParagraph* para1 = GetParagraphAtPosition(range.GetStart());
1823 wxRichTextParagraph* para2 = GetParagraphAtPosition(range.GetEnd());
1824 if (para1)
1825 range.SetStart(para1->GetRange().GetStart());
1826 if (para2)
1827 range.SetEnd(para2->GetRange().GetEnd());
1828 }
1829 return range;
1830}
1831
5d7836c4
JS
1832/*!
1833 * wxRichTextFragment class declaration
1834 * This is a lind of paragraph layout box used for storing
1835 * paragraphs for Undo/Redo, for example.
1836 */
1837
1838IMPLEMENT_DYNAMIC_CLASS(wxRichTextFragment, wxRichTextParagraphLayoutBox)
1839
1840/// Initialise
1841void wxRichTextFragment::Init()
1842{
1843 m_partialParagraph = false;
1844}
1845
1846/// Copy
1847void wxRichTextFragment::Copy(const wxRichTextFragment& obj)
1848{
1849 wxRichTextParagraphLayoutBox::Copy(obj);
1850
1851 m_partialParagraph = obj.m_partialParagraph;
1852}
1853
1854/*!
1855 * wxRichTextParagraph
1856 * This object represents a single paragraph (or in a straight text editor, a line).
1857 */
1858
1859IMPLEMENT_DYNAMIC_CLASS(wxRichTextParagraph, wxRichTextBox)
1860
1861wxRichTextParagraph::wxRichTextParagraph(wxRichTextObject* parent, wxTextAttrEx* style):
1862 wxRichTextBox(parent)
1863{
1864 if (parent && !style)
1865 SetAttributes(parent->GetAttributes());
1866 if (style)
1867 SetAttributes(*style);
1868}
1869
1870wxRichTextParagraph::wxRichTextParagraph(const wxString& text, wxRichTextObject* parent, wxTextAttrEx* style):
1871 wxRichTextBox(parent)
1872{
1873 if (parent && !style)
1874 SetAttributes(parent->GetAttributes());
1875 if (style)
1876 SetAttributes(*style);
1877
1878 AppendChild(new wxRichTextPlainText(text, this));
1879}
1880
1881wxRichTextParagraph::~wxRichTextParagraph()
1882{
1883 ClearLines();
1884}
1885
1886/// Draw the item
1887bool wxRichTextParagraph::Draw(wxDC& dc, const wxRichTextRange& WXUNUSED(range), const wxRichTextRange& selectionRange, const wxRect& WXUNUSED(rect), int WXUNUSED(descent), int style)
1888{
1889 // Draw the bullet, if any
1890 if (GetAttributes().GetBulletStyle() != wxTEXT_ATTR_BULLET_STYLE_NONE)
1891 {
1892 if (GetAttributes().GetLeftSubIndent() != 0)
1893 {
1894 int spaceBeforePara = ConvertTenthsMMToPixels(dc, GetAttributes().GetParagraphSpacingBefore());
1895 // int spaceAfterPara = ConvertTenthsMMToPixels(dc, GetAttributes().GetParagraphSpacingAfter());
1896 int leftIndent = ConvertTenthsMMToPixels(dc, GetAttributes().GetLeftIndent());
1897 // int leftSubIndent = ConvertTenthsMMToPixels(dc, GetAttributes().GetLeftSubIndent());
1898 // int rightIndent = ConvertTenthsMMToPixels(dc, GetAttributes().GetRightIndent());
1899
1900 if (GetAttributes().GetBulletStyle() & wxTEXT_ATTR_BULLET_STYLE_BITMAP)
1901 {
1902 // TODO
1903 }
1904 else
1905 {
1906 wxString bulletText = GetBulletText();
7fe8059f 1907 if (!bulletText.empty())
5d7836c4
JS
1908 {
1909 if (GetAttributes().GetFont().Ok())
1910 dc.SetFont(GetAttributes().GetFont());
7fe8059f 1911
5d7836c4
JS
1912 if (GetAttributes().GetTextColour().Ok())
1913 dc.SetTextForeground(GetAttributes().GetTextColour());
7fe8059f 1914
5d7836c4 1915 dc.SetBackgroundMode(wxTRANSPARENT);
7fe8059f 1916
5d7836c4
JS
1917 // Get line height from first line, if any
1918 wxRichTextLine* line = m_cachedLines.GetFirst() ? (wxRichTextLine* ) m_cachedLines.GetFirst()->GetData() : (wxRichTextLine*) NULL;
7fe8059f 1919
5d7836c4 1920 wxPoint linePos;
4e09ebe8 1921 int lineHeight wxDUMMY_INITIALIZE(0);
5d7836c4
JS
1922 if (line)
1923 {
1924 lineHeight = line->GetSize().y;
1925 linePos = line->GetPosition() + GetPosition();
1926 }
1927 else
1928 {
1929 lineHeight = dc.GetCharHeight();
1930 linePos = GetPosition();
1931 linePos.y += spaceBeforePara;
1932 }
7fe8059f 1933
5d7836c4 1934 int charHeight = dc.GetCharHeight();
7fe8059f 1935
5d7836c4
JS
1936 int x = GetPosition().x + leftIndent;
1937 int y = linePos.y + (lineHeight - charHeight);
7fe8059f 1938
5d7836c4
JS
1939 dc.DrawText(bulletText, x, y);
1940 }
1941 }
1942 }
1943 }
7fe8059f 1944
5d7836c4
JS
1945 // Draw the range for each line, one object at a time.
1946
1947 wxRichTextLineList::compatibility_iterator node = m_cachedLines.GetFirst();
1948 while (node)
1949 {
1950 wxRichTextLine* line = node->GetData();
1e967276 1951 wxRichTextRange lineRange = line->GetAbsoluteRange();
5d7836c4
JS
1952
1953 int maxDescent = line->GetDescent();
1954
1955 // Lines are specified relative to the paragraph
1956
1957 wxPoint linePosition = line->GetPosition() + GetPosition();
1958 wxPoint objectPosition = linePosition;
1959
1960 // Loop through objects until we get to the one within range
1961 wxRichTextObjectList::compatibility_iterator node2 = m_children.GetFirst();
1962 while (node2)
1963 {
1964 wxRichTextObject* child = node2->GetData();
1e967276 1965 if (!child->GetRange().IsOutside(lineRange))
5d7836c4
JS
1966 {
1967 // Draw this part of the line at the correct position
1968 wxRichTextRange objectRange(child->GetRange());
1e967276 1969 objectRange.LimitTo(lineRange);
5d7836c4
JS
1970
1971 wxSize objectSize;
1972 int descent = 0;
7f0d9d71 1973 child->GetRangeSize(objectRange, objectSize, descent, dc, wxRICHTEXT_UNFORMATTED, objectPosition);
5d7836c4
JS
1974
1975 // Use the child object's width, but the whole line's height
1976 wxRect childRect(objectPosition, wxSize(objectSize.x, line->GetSize().y));
1977 child->Draw(dc, objectRange, selectionRange, childRect, maxDescent, style);
1978
1979 objectPosition.x += objectSize.x;
1980 }
1e967276 1981 else if (child->GetRange().GetStart() > lineRange.GetEnd())
5d7836c4
JS
1982 // Can break out of inner loop now since we've passed this line's range
1983 break;
1984
1985 node2 = node2->GetNext();
1986 }
1987
1988 node = node->GetNext();
7fe8059f 1989 }
5d7836c4
JS
1990
1991 return true;
1992}
1993
1994/// Lay the item out
38113684 1995bool wxRichTextParagraph::Layout(wxDC& dc, const wxRect& rect, int style)
5d7836c4 1996{
169adfa9
JS
1997 // ClearLines();
1998
5d7836c4
JS
1999 // Increase the size of the paragraph due to spacing
2000 int spaceBeforePara = ConvertTenthsMMToPixels(dc, GetAttributes().GetParagraphSpacingBefore());
2001 int spaceAfterPara = ConvertTenthsMMToPixels(dc, GetAttributes().GetParagraphSpacingAfter());
2002 int leftIndent = ConvertTenthsMMToPixels(dc, GetAttributes().GetLeftIndent());
2003 int leftSubIndent = ConvertTenthsMMToPixels(dc, GetAttributes().GetLeftSubIndent());
2004 int rightIndent = ConvertTenthsMMToPixels(dc, GetAttributes().GetRightIndent());
2005
2006 int lineSpacing = 0;
2007
2008 // Let's assume line spacing of 10 is normal, 15 is 1.5, 20 is 2, etc.
2009 if (GetAttributes().GetLineSpacing() > 10 && GetAttributes().GetFont().Ok())
2010 {
2011 dc.SetFont(GetAttributes().GetFont());
2012 lineSpacing = (ConvertTenthsMMToPixels(dc, dc.GetCharHeight()) * GetAttributes().GetLineSpacing())/10;
2013 }
2014
2015 // Available space for text on each line differs.
2016 int availableTextSpaceFirstLine = rect.GetWidth() - leftIndent - rightIndent;
2017
2018 // Bullets start the text at the same position as subsequent lines
2019 if (GetAttributes().GetBulletStyle() != wxTEXT_ATTR_BULLET_STYLE_NONE)
2020 availableTextSpaceFirstLine -= leftSubIndent;
2021
2022 int availableTextSpaceSubsequentLines = rect.GetWidth() - leftIndent - rightIndent - leftSubIndent;
2023
2024 // Start position for each line relative to the paragraph
2025 int startPositionFirstLine = leftIndent;
2026 int startPositionSubsequentLines = leftIndent + leftSubIndent;
2027
2028 // If we have a bullet in this paragraph, the start position for the first line's text
2029 // is actually leftIndent + leftSubIndent.
2030 if (GetAttributes().GetBulletStyle() != wxTEXT_ATTR_BULLET_STYLE_NONE)
2031 startPositionFirstLine = startPositionSubsequentLines;
2032
2033 //bool restrictWidth = wxRichTextHasStyle(style, wxRICHTEXT_FIXED_WIDTH);
2034 //bool restrictHeight = wxRichTextHasStyle(style, wxRICHTEXT_FIXED_HEIGHT);
2035
2036 long lastEndPos = GetRange().GetStart()-1;
2037 long lastCompletedEndPos = lastEndPos;
2038
2039 int currentWidth = 0;
2040 SetPosition(rect.GetPosition());
2041
2042 wxPoint currentPosition(0, spaceBeforePara); // We will calculate lines relative to paragraph
2043 int lineHeight = 0;
2044 int maxWidth = 0;
2045 int maxDescent = 0;
2046
2047 int lineCount = 0;
2048
2049 // Split up lines
2050
2051 // We may need to go back to a previous child, in which case create the new line,
2052 // find the child corresponding to the start position of the string, and
2053 // continue.
2054
2055 wxRichTextObjectList::compatibility_iterator node = m_children.GetFirst();
2056 while (node)
2057 {
2058 wxRichTextObject* child = node->GetData();
2059
2060 // If this is e.g. a composite text box, it will need to be laid out itself.
2061 // But if just a text fragment or image, for example, this will
2062 // do nothing. NB: won't we need to set the position after layout?
2063 // since for example if position is dependent on vertical line size, we
2064 // can't tell the position until the size is determined. So possibly introduce
2065 // another layout phase.
2066
38113684 2067 child->Layout(dc, rect, style);
5d7836c4
JS
2068
2069 // Available width depends on whether we're on the first or subsequent lines
2070 int availableSpaceForText = (lineCount == 0 ? availableTextSpaceFirstLine : availableTextSpaceSubsequentLines);
2071
2072 currentPosition.x = (lineCount == 0 ? startPositionFirstLine : startPositionSubsequentLines);
2073
2074 // We may only be looking at part of a child, if we searched back for wrapping
2075 // and found a suitable point some way into the child. So get the size for the fragment
2076 // if necessary.
2077
2078 wxSize childSize;
2079 int childDescent = 0;
2080 if (lastEndPos == child->GetRange().GetStart() - 1)
2081 {
2082 childSize = child->GetCachedSize();
2083 childDescent = child->GetDescent();
2084 }
2085 else
7f0d9d71 2086 GetRangeSize(wxRichTextRange(lastEndPos+1, child->GetRange().GetEnd()), childSize, childDescent, dc, wxRICHTEXT_UNFORMATTED,rect.GetPosition());
5d7836c4
JS
2087
2088 if (childSize.x + currentWidth > availableSpaceForText)
2089 {
2090 long wrapPosition = 0;
2091
2092 // Find a place to wrap. This may walk back to previous children,
2093 // for example if a word spans several objects.
2094 if (!FindWrapPosition(wxRichTextRange(lastCompletedEndPos+1, child->GetRange().GetEnd()), dc, availableSpaceForText, wrapPosition))
2095 {
2096 // If the function failed, just cut it off at the end of this child.
2097 wrapPosition = child->GetRange().GetEnd();
2098 }
2099
2100 // FindWrapPosition can still return a value that will put us in an endless wrapping loop
2101 if (wrapPosition <= lastCompletedEndPos)
2102 wrapPosition = wxMax(lastCompletedEndPos+1,child->GetRange().GetEnd());
2103
2104 // wxLogDebug(wxT("Split at %ld"), wrapPosition);
7fe8059f 2105
5d7836c4
JS
2106 // Let's find the actual size of the current line now
2107 wxSize actualSize;
2108 wxRichTextRange actualRange(lastCompletedEndPos+1, wrapPosition);
2109 GetRangeSize(actualRange, actualSize, childDescent, dc, wxRICHTEXT_UNFORMATTED);
2110 currentWidth = actualSize.x;
2111 lineHeight = wxMax(lineHeight, actualSize.y);
2112 maxDescent = wxMax(childDescent, maxDescent);
7fe8059f 2113
5d7836c4 2114 // Add a new line
1e967276 2115 wxRichTextLine* line = AllocateLine(lineCount);
39a1c2f2 2116
1e967276
JS
2117 // Set relative range so we won't have to change line ranges when paragraphs are moved
2118 line->SetRange(wxRichTextRange(actualRange.GetStart() - GetRange().GetStart(), actualRange.GetEnd() - GetRange().GetStart()));
5d7836c4
JS
2119 line->SetPosition(currentPosition);
2120 line->SetSize(wxSize(currentWidth, lineHeight));
2121 line->SetDescent(maxDescent);
2122
5d7836c4
JS
2123 // Now move down a line. TODO: add margins, spacing
2124 currentPosition.y += lineHeight;
2125 currentPosition.y += lineSpacing;
2126 currentWidth = 0;
2127 maxDescent = 0;
7fe8059f
WS
2128 maxWidth = wxMax(maxWidth, currentWidth);
2129
5d7836c4
JS
2130 lineCount ++;
2131
2132 // TODO: account for zero-length objects, such as fields
2133 wxASSERT(wrapPosition > lastCompletedEndPos);
7fe8059f 2134
5d7836c4
JS
2135 lastEndPos = wrapPosition;
2136 lastCompletedEndPos = lastEndPos;
2137
2138 lineHeight = 0;
2139
2140 // May need to set the node back to a previous one, due to searching back in wrapping
2141 wxRichTextObject* childAfterWrapPosition = FindObjectAtPosition(wrapPosition+1);
2142 if (childAfterWrapPosition)
2143 node = m_children.Find(childAfterWrapPosition);
2144 else
2145 node = node->GetNext();
2146 }
2147 else
2148 {
2149 // We still fit, so don't add a line, and keep going
2150 currentWidth += childSize.x;
2151 lineHeight = wxMax(lineHeight, childSize.y);
2152 maxDescent = wxMax(childDescent, maxDescent);
2153
2154 maxWidth = wxMax(maxWidth, currentWidth);
2155 lastEndPos = child->GetRange().GetEnd();
2156
2157 node = node->GetNext();
2158 }
2159 }
2160
2161 // Add the last line - it's the current pos -> last para pos
2162 // Substract -1 because the last position is always the end-paragraph position.
2163 if (lastCompletedEndPos <= GetRange().GetEnd()-1)
2164 {
2165 currentPosition.x = (lineCount == 0 ? startPositionFirstLine : startPositionSubsequentLines);
2166
1e967276
JS
2167 wxRichTextLine* line = AllocateLine(lineCount);
2168
2169 wxRichTextRange actualRange(lastCompletedEndPos+1, GetRange().GetEnd()-1);
2170
2171 // Set relative range so we won't have to change line ranges when paragraphs are moved
2172 line->SetRange(wxRichTextRange(actualRange.GetStart() - GetRange().GetStart(), actualRange.GetEnd() - GetRange().GetStart()));
7fe8059f 2173
5d7836c4
JS
2174 line->SetPosition(currentPosition);
2175
2176 if (lineHeight == 0)
2177 {
2178 if (GetAttributes().GetFont().Ok())
2179 dc.SetFont(GetAttributes().GetFont());
2180 lineHeight = dc.GetCharHeight();
2181 }
2182 if (maxDescent == 0)
2183 {
2184 int w, h;
2185 dc.GetTextExtent(wxT("X"), & w, &h, & maxDescent);
2186 }
2187
2188 line->SetSize(wxSize(currentWidth, lineHeight));
2189 line->SetDescent(maxDescent);
2190 currentPosition.y += lineHeight;
2191 currentPosition.y += lineSpacing;
2192 lineCount ++;
5d7836c4
JS
2193 }
2194
1e967276
JS
2195 // Remove remaining unused line objects, if any
2196 ClearUnusedLines(lineCount);
2197
5d7836c4
JS
2198 // Apply styles to wrapped lines
2199 ApplyParagraphStyle(rect);
2200
2201 SetCachedSize(wxSize(maxWidth, currentPosition.y + spaceBeforePara + spaceAfterPara));
2202
2203 m_dirty = false;
2204
2205 return true;
2206}
2207
2208/// Apply paragraph styles, such as centering, to wrapped lines
2209void wxRichTextParagraph::ApplyParagraphStyle(const wxRect& rect)
2210{
2211 if (!GetAttributes().HasAlignment())
2212 return;
2213
2214 wxRichTextLineList::compatibility_iterator node = m_cachedLines.GetFirst();
2215 while (node)
2216 {
2217 wxRichTextLine* line = node->GetData();
2218
2219 wxPoint pos = line->GetPosition();
2220 wxSize size = line->GetSize();
2221
2222 // centering, right-justification
2223 if (GetAttributes().HasAlignment() && GetAttributes().GetAlignment() == wxTEXT_ALIGNMENT_CENTRE)
2224 {
2225 pos.x = (rect.GetWidth() - size.x)/2 + pos.x;
2226 line->SetPosition(pos);
2227 }
2228 else if (GetAttributes().HasAlignment() && GetAttributes().GetAlignment() == wxTEXT_ALIGNMENT_RIGHT)
2229 {
2230 pos.x = rect.GetRight() - size.x;
2231 line->SetPosition(pos);
2232 }
2233
2234 node = node->GetNext();
2235 }
2236}
2237
2238/// Insert text at the given position
2239bool wxRichTextParagraph::InsertText(long pos, const wxString& text)
2240{
2241 wxRichTextObject* childToUse = NULL;
09f14108 2242 wxRichTextObjectList::compatibility_iterator nodeToUse = wxRichTextObjectList::compatibility_iterator();
5d7836c4
JS
2243
2244 wxRichTextObjectList::compatibility_iterator node = m_children.GetFirst();
2245 while (node)
2246 {
2247 wxRichTextObject* child = node->GetData();
2248 if (child->GetRange().Contains(pos) && child->GetRange().GetLength() > 0)
2249 {
2250 childToUse = child;
2251 nodeToUse = node;
2252 break;
2253 }
2254
2255 node = node->GetNext();
2256 }
2257
2258 if (childToUse)
2259 {
2260 wxRichTextPlainText* textObject = wxDynamicCast(childToUse, wxRichTextPlainText);
2261 if (textObject)
2262 {
2263 int posInString = pos - textObject->GetRange().GetStart();
2264
2265 wxString newText = textObject->GetText().Mid(0, posInString) +
2266 text + textObject->GetText().Mid(posInString);
2267 textObject->SetText(newText);
2268
28f92d74 2269 int textLength = text.length();
5d7836c4
JS
2270
2271 textObject->SetRange(wxRichTextRange(textObject->GetRange().GetStart(),
2272 textObject->GetRange().GetEnd() + textLength));
2273
2274 // Increment the end range of subsequent fragments in this paragraph.
2275 // We'll set the paragraph range itself at a higher level.
2276
2277 wxRichTextObjectList::compatibility_iterator node = nodeToUse->GetNext();
2278 while (node)
2279 {
2280 wxRichTextObject* child = node->GetData();
2281 child->SetRange(wxRichTextRange(textObject->GetRange().GetStart() + textLength,
2282 textObject->GetRange().GetEnd() + textLength));
7fe8059f 2283
5d7836c4
JS
2284 node = node->GetNext();
2285 }
2286
2287 return true;
2288 }
2289 else
2290 {
2291 // TODO: if not a text object, insert at closest position, e.g. in front of it
2292 }
2293 }
2294 else
2295 {
2296 // Add at end.
2297 // Don't pass parent initially to suppress auto-setting of parent range.
2298 // We'll do that at a higher level.
2299 wxRichTextPlainText* textObject = new wxRichTextPlainText(text, this);
2300
2301 AppendChild(textObject);
2302 return true;
2303 }
2304
2305 return false;
2306}
2307
2308void wxRichTextParagraph::Copy(const wxRichTextParagraph& obj)
2309{
2310 wxRichTextBox::Copy(obj);
2311}
2312
2313/// Clear the cached lines
2314void wxRichTextParagraph::ClearLines()
2315{
2316 WX_CLEAR_LIST(wxRichTextLineList, m_cachedLines);
2317}
2318
2319/// Get/set the object size for the given range. Returns false if the range
2320/// is invalid for this object.
7f0d9d71 2321bool wxRichTextParagraph::GetRangeSize(const wxRichTextRange& range, wxSize& size, int& descent, wxDC& dc, int flags, wxPoint position) const
5d7836c4
JS
2322{
2323 if (!range.IsWithin(GetRange()))
2324 return false;
2325
2326 if (flags & wxRICHTEXT_UNFORMATTED)
2327 {
2328 // Just use unformatted data, assume no line breaks
2329 // TODO: take into account line breaks
2330
2331 wxSize sz;
2332
2333 wxRichTextObjectList::compatibility_iterator node = m_children.GetFirst();
2334 while (node)
2335 {
2336 wxRichTextObject* child = node->GetData();
2337 if (!child->GetRange().IsOutside(range))
2338 {
2339 wxSize childSize;
7fe8059f 2340
5d7836c4
JS
2341 wxRichTextRange rangeToUse = range;
2342 rangeToUse.LimitTo(child->GetRange());
2343 int childDescent = 0;
7fe8059f 2344
7f0d9d71 2345 if (child->GetRangeSize(rangeToUse, childSize, childDescent, dc, flags, position))
5d7836c4
JS
2346 {
2347 sz.y = wxMax(sz.y, childSize.y);
2348 sz.x += childSize.x;
2349 descent = wxMax(descent, childDescent);
2350 }
2351 }
2352
2353 node = node->GetNext();
2354 }
2355 size = sz;
2356 }
2357 else
2358 {
2359 // Use formatted data, with line breaks
2360 wxSize sz;
2361
2362 // We're going to loop through each line, and then for each line,
2363 // call GetRangeSize for the fragment that comprises that line.
2364 // Only we have to do that multiple times within the line, because
2365 // the line may be broken into pieces. For now ignore line break commands
2366 // (so we can assume that getting the unformatted size for a fragment
2367 // within a line is the actual size)
2368
2369 wxRichTextLineList::compatibility_iterator node = m_cachedLines.GetFirst();
2370 while (node)
2371 {
2372 wxRichTextLine* line = node->GetData();
1e967276
JS
2373 wxRichTextRange lineRange = line->GetAbsoluteRange();
2374 if (!lineRange.IsOutside(range))
5d7836c4
JS
2375 {
2376 wxSize lineSize;
7fe8059f 2377
5d7836c4
JS
2378 wxRichTextObjectList::compatibility_iterator node2 = m_children.GetFirst();
2379 while (node2)
2380 {
2381 wxRichTextObject* child = node2->GetData();
7fe8059f 2382
1e967276 2383 if (!child->GetRange().IsOutside(lineRange))
5d7836c4 2384 {
1e967276 2385 wxRichTextRange rangeToUse = lineRange;
5d7836c4 2386 rangeToUse.LimitTo(child->GetRange());
7fe8059f 2387
5d7836c4
JS
2388 wxSize childSize;
2389 int childDescent = 0;
7f0d9d71 2390 if (child->GetRangeSize(rangeToUse, childSize, childDescent, dc, flags, position))
5d7836c4
JS
2391 {
2392 lineSize.y = wxMax(lineSize.y, childSize.y);
2393 lineSize.x += childSize.x;
2394 }
2395 descent = wxMax(descent, childDescent);
2396 }
7fe8059f 2397
5d7836c4
JS
2398 node2 = node2->GetNext();
2399 }
2400
2401 // Increase size by a line (TODO: paragraph spacing)
2402 sz.y += lineSize.y;
2403 sz.x = wxMax(sz.x, lineSize.x);
2404 }
2405 node = node->GetNext();
2406 }
2407 size = sz;
2408 }
2409 return true;
2410}
2411
2412/// Finds the absolute position and row height for the given character position
2413bool wxRichTextParagraph::FindPosition(wxDC& dc, long index, wxPoint& pt, int* height, bool forceLineStart)
2414{
2415 if (index == -1)
2416 {
2417 wxRichTextLine* line = ((wxRichTextParagraphLayoutBox*)GetParent())->GetLineAtPosition(0);
2418 if (line)
2419 *height = line->GetSize().y;
2420 else
2421 *height = dc.GetCharHeight();
2422
2423 // -1 means 'the start of the buffer'.
2424 pt = GetPosition();
2425 if (line)
2426 pt = pt + line->GetPosition();
2427
2428 *height = dc.GetCharHeight();
2429
2430 return true;
2431 }
2432
2433 // The final position in a paragraph is taken to mean the position
2434 // at the start of the next paragraph.
2435 if (index == GetRange().GetEnd())
2436 {
2437 wxRichTextParagraphLayoutBox* parent = wxDynamicCast(GetParent(), wxRichTextParagraphLayoutBox);
2438 wxASSERT( parent != NULL );
2439
2440 // Find the height at the next paragraph, if any
2441 wxRichTextLine* line = parent->GetLineAtPosition(index + 1);
2442 if (line)
2443 {
2444 *height = line->GetSize().y;
2445 pt = line->GetAbsolutePosition();
2446 }
2447 else
2448 {
2449 *height = dc.GetCharHeight();
2450 int indent = ConvertTenthsMMToPixels(dc, m_attributes.GetLeftIndent());
2451 pt = wxPoint(indent, GetCachedSize().y);
2452 }
2453
2454 return true;
2455 }
2456
2457 if (index < GetRange().GetStart() || index > GetRange().GetEnd())
2458 return false;
2459
2460 wxRichTextLineList::compatibility_iterator node = m_cachedLines.GetFirst();
2461 while (node)
2462 {
2463 wxRichTextLine* line = node->GetData();
1e967276
JS
2464 wxRichTextRange lineRange = line->GetAbsoluteRange();
2465 if (index >= lineRange.GetStart() && index <= lineRange.GetEnd())
5d7836c4
JS
2466 {
2467 // If this is the last point in the line, and we're forcing the
2468 // returned value to be the start of the next line, do the required
2469 // thing.
1e967276 2470 if (index == lineRange.GetEnd() && forceLineStart)
5d7836c4
JS
2471 {
2472 if (node->GetNext())
2473 {
2474 wxRichTextLine* nextLine = node->GetNext()->GetData();
2475 *height = nextLine->GetSize().y;
2476 pt = nextLine->GetAbsolutePosition();
2477 return true;
2478 }
2479 }
2480
2481 pt.y = line->GetPosition().y + GetPosition().y;
2482
1e967276 2483 wxRichTextRange r(lineRange.GetStart(), index);
5d7836c4
JS
2484 wxSize rangeSize;
2485 int descent = 0;
2486
2487 // We find the size of the line up to this point,
2488 // then we can add this size to the line start position and
2489 // paragraph start position to find the actual position.
2490
7f0d9d71 2491 if (GetRangeSize(r, rangeSize, descent, dc, wxRICHTEXT_UNFORMATTED, line->GetPosition()+ GetPosition()))
5d7836c4
JS
2492 {
2493 pt.x = line->GetPosition().x + GetPosition().x + rangeSize.x;
2494 *height = line->GetSize().y;
2495
2496 return true;
2497 }
2498
2499 }
2500
2501 node = node->GetNext();
2502 }
2503
2504 return false;
2505}
2506
2507/// Hit-testing: returns a flag indicating hit test details, plus
2508/// information about position
2509int wxRichTextParagraph::HitTest(wxDC& dc, const wxPoint& pt, long& textPosition)
2510{
2511 wxPoint paraPos = GetPosition();
2512
2513 wxRichTextLineList::compatibility_iterator node = m_cachedLines.GetFirst();
2514 while (node)
2515 {
2516 wxRichTextLine* line = node->GetData();
2517 wxPoint linePos = paraPos + line->GetPosition();
2518 wxSize lineSize = line->GetSize();
1e967276 2519 wxRichTextRange lineRange = line->GetAbsoluteRange();
5d7836c4
JS
2520
2521 if (pt.y >= linePos.y && pt.y <= linePos.y + lineSize.y)
2522 {
2523 if (pt.x < linePos.x)
2524 {
1e967276 2525 textPosition = lineRange.GetStart();
5d7836c4
JS
2526 return wxRICHTEXT_HITTEST_BEFORE;
2527 }
2528 else if (pt.x >= (linePos.x + lineSize.x))
2529 {
1e967276 2530 textPosition = lineRange.GetEnd();
5d7836c4
JS
2531 return wxRICHTEXT_HITTEST_AFTER;
2532 }
2533 else
2534 {
2535 long i;
2536 int lastX = linePos.x;
1e967276 2537 for (i = lineRange.GetStart(); i <= lineRange.GetEnd(); i++)
5d7836c4
JS
2538 {
2539 wxSize childSize;
2540 int descent = 0;
7fe8059f 2541
1e967276 2542 wxRichTextRange rangeToUse(lineRange.GetStart(), i);
7fe8059f 2543
7f0d9d71 2544 GetRangeSize(rangeToUse, childSize, descent, dc, wxRICHTEXT_UNFORMATTED, linePos);
5d7836c4
JS
2545
2546 int nextX = childSize.x + linePos.x;
2547
2548 if (pt.x >= lastX && pt.x <= nextX)
2549 {
2550 textPosition = i;
2551
2552 // So now we know it's between i-1 and i.
2553 // Let's see if we can be more precise about
2554 // which side of the position it's on.
2555
2556 int midPoint = (nextX - lastX)/2 + lastX;
2557 if (pt.x >= midPoint)
2558 return wxRICHTEXT_HITTEST_AFTER;
2559 else
2560 return wxRICHTEXT_HITTEST_BEFORE;
2561 }
2562 else
2563 {
2564 lastX = nextX;
2565 }
2566 }
2567 }
2568 }
7fe8059f 2569
5d7836c4
JS
2570 node = node->GetNext();
2571 }
2572
2573 return wxRICHTEXT_HITTEST_NONE;
2574}
2575
2576/// Split an object at this position if necessary, and return
2577/// the previous object, or NULL if inserting at beginning.
2578wxRichTextObject* wxRichTextParagraph::SplitAt(long pos, wxRichTextObject** previousObject)
2579{
2580 wxRichTextObjectList::compatibility_iterator node = m_children.GetFirst();
2581 while (node)
2582 {
2583 wxRichTextObject* child = node->GetData();
2584
2585 if (pos == child->GetRange().GetStart())
2586 {
2587 if (previousObject)
4d551ad5
JS
2588 {
2589 if (node->GetPrevious())
2590 *previousObject = node->GetPrevious()->GetData();
2591 else
2592 *previousObject = NULL;
2593 }
5d7836c4
JS
2594
2595 return child;
2596 }
2597
2598 if (child->GetRange().Contains(pos))
2599 {
2600 // This should create a new object, transferring part of
2601 // the content to the old object and the rest to the new object.
2602 wxRichTextObject* newObject = child->DoSplit(pos);
2603
2604 // If we couldn't split this object, just insert in front of it.
2605 if (!newObject)
2606 {
2607 // Maybe this is an empty string, try the next one
2608 // return child;
2609 }
2610 else
2611 {
2612 // Insert the new object after 'child'
2613 if (node->GetNext())
2614 m_children.Insert(node->GetNext(), newObject);
2615 else
2616 m_children.Append(newObject);
2617 newObject->SetParent(this);
2618
2619 if (previousObject)
2620 *previousObject = child;
2621
2622 return newObject;
2623 }
2624 }
2625
2626 node = node->GetNext();
2627 }
2628 if (previousObject)
2629 *previousObject = NULL;
2630 return NULL;
2631}
2632
2633/// Move content to a list from obj on
2634void wxRichTextParagraph::MoveToList(wxRichTextObject* obj, wxList& list)
2635{
2636 wxRichTextObjectList::compatibility_iterator node = m_children.Find(obj);
2637 while (node)
2638 {
2639 wxRichTextObject* child = node->GetData();
2640 list.Append(child);
2641
2642 wxRichTextObjectList::compatibility_iterator oldNode = node;
2643
2644 node = node->GetNext();
2645
2646 m_children.DeleteNode(oldNode);
2647 }
2648}
2649
2650/// Add content back from list
2651void wxRichTextParagraph::MoveFromList(wxList& list)
2652{
09f14108 2653 for (wxList::compatibility_iterator node = list.GetFirst(); node; node = node->GetNext())
5d7836c4
JS
2654 {
2655 AppendChild((wxRichTextObject*) node->GetData());
2656 }
2657}
2658
2659/// Calculate range
2660void wxRichTextParagraph::CalculateRange(long start, long& end)
2661{
2662 wxRichTextCompositeObject::CalculateRange(start, end);
2663
2664 // Add one for end of paragraph
2665 end ++;
2666
2667 m_range.SetRange(start, end);
2668}
2669
2670/// Find the object at the given position
2671wxRichTextObject* wxRichTextParagraph::FindObjectAtPosition(long position)
2672{
2673 wxRichTextObjectList::compatibility_iterator node = m_children.GetFirst();
2674 while (node)
2675 {
2676 wxRichTextObject* obj = node->GetData();
2677 if (obj->GetRange().Contains(position))
2678 return obj;
7fe8059f 2679
5d7836c4
JS
2680 node = node->GetNext();
2681 }
2682 return NULL;
2683}
2684
2685/// Get the plain text searching from the start or end of the range.
2686/// The resulting string may be shorter than the range given.
2687bool wxRichTextParagraph::GetContiguousPlainText(wxString& text, const wxRichTextRange& range, bool fromStart)
2688{
2689 text = wxEmptyString;
2690
2691 if (fromStart)
2692 {
2693 wxRichTextObjectList::compatibility_iterator node = m_children.GetFirst();
2694 while (node)
2695 {
2696 wxRichTextObject* obj = node->GetData();
2697 if (!obj->GetRange().IsOutside(range))
2698 {
2699 wxRichTextPlainText* textObj = wxDynamicCast(obj, wxRichTextPlainText);
2700 if (textObj)
2701 {
2702 text += textObj->GetTextForRange(range);
2703 }
2704 else
2705 return true;
2706 }
2707
2708 node = node->GetNext();
2709 }
2710 }
2711 else
2712 {
2713 wxRichTextObjectList::compatibility_iterator node = m_children.GetLast();
2714 while (node)
2715 {
2716 wxRichTextObject* obj = node->GetData();
2717 if (!obj->GetRange().IsOutside(range))
2718 {
2719 wxRichTextPlainText* textObj = wxDynamicCast(obj, wxRichTextPlainText);
2720 if (textObj)
2721 {
2722 text = textObj->GetTextForRange(range) + text;
2723 }
2724 else
2725 return true;
2726 }
2727
2728 node = node->GetPrevious();
2729 }
2730 }
2731
2732 return true;
2733}
2734
2735/// Find a suitable wrap position.
2736bool wxRichTextParagraph::FindWrapPosition(const wxRichTextRange& range, wxDC& dc, int availableSpace, long& wrapPosition)
2737{
2738 // Find the first position where the line exceeds the available space.
2739 wxSize sz;
2740 long i;
2741 long breakPosition = range.GetEnd();
2742 for (i = range.GetStart(); i <= range.GetEnd(); i++)
2743 {
2744 int descent = 0;
2745 GetRangeSize(wxRichTextRange(range.GetStart(), i), sz, descent, dc, wxRICHTEXT_UNFORMATTED);
2746
2747 if (sz.x > availableSpace)
2748 {
2749 breakPosition = i-1;
2750 break;
2751 }
2752 }
2753
2754 // Now we know the last position on the line.
2755 // Let's try to find a word break.
2756
2757 wxString plainText;
2758 if (GetContiguousPlainText(plainText, wxRichTextRange(range.GetStart(), breakPosition), false))
2759 {
2760 int spacePos = plainText.Find(wxT(' '), true);
2761 if (spacePos != wxNOT_FOUND)
2762 {
28f92d74 2763 int positionsFromEndOfString = plainText.length() - spacePos - 1;
5d7836c4
JS
2764 breakPosition = breakPosition - positionsFromEndOfString;
2765 }
2766 }
2767
2768 wrapPosition = breakPosition;
2769
2770 return true;
2771}
2772
2773/// Get the bullet text for this paragraph.
2774wxString wxRichTextParagraph::GetBulletText()
2775{
2776 if (GetAttributes().GetBulletStyle() == wxTEXT_ATTR_BULLET_STYLE_NONE ||
2777 (GetAttributes().GetBulletStyle() & wxTEXT_ATTR_BULLET_STYLE_BITMAP))
2778 return wxEmptyString;
2779
2780 int number = GetAttributes().GetBulletNumber();
2781
2782 wxString text;
2783 if (GetAttributes().GetBulletStyle() & wxTEXT_ATTR_BULLET_STYLE_ARABIC)
2784 {
2785 text.Printf(wxT("%d"), number);
2786 }
2787 else if (GetAttributes().GetBulletStyle() & wxTEXT_ATTR_BULLET_STYLE_LETTERS_UPPER)
2788 {
2789 // TODO: Unicode, and also check if number > 26
2790 text.Printf(wxT("%c"), (wxChar) (number+64));
2791 }
2792 else if (GetAttributes().GetBulletStyle() & wxTEXT_ATTR_BULLET_STYLE_LETTERS_LOWER)
2793 {
2794 // TODO: Unicode, and also check if number > 26
2795 text.Printf(wxT("%c"), (wxChar) (number+96));
2796 }
2797 else if (GetAttributes().GetBulletStyle() & wxTEXT_ATTR_BULLET_STYLE_ROMAN_UPPER)
2798 {
2799 // TODO: convert from number to roman numeral
2800 if (number == 1)
2801 text = wxT("I");
2802 else if (number == 2)
2803 text = wxT("II");
2804 else if (number == 3)
2805 text = wxT("III");
2806 else if (number == 4)
2807 text = wxT("IV");
2808 else
2809 text = wxT("TODO");
2810 }
2811 else if (GetAttributes().GetBulletStyle() & wxTEXT_ATTR_BULLET_STYLE_ROMAN_LOWER)
2812 {
2813 // TODO: convert from number to roman numeral
2814 if (number == 1)
2815 text = wxT("i");
2816 else if (number == 2)
2817 text = wxT("ii");
2818 else if (number == 3)
2819 text = wxT("iii");
2820 else if (number == 4)
2821 text = wxT("iv");
2822 else
2823 text = wxT("TODO");
2824 }
2825 else if (GetAttributes().GetBulletStyle() & wxTEXT_ATTR_BULLET_STYLE_SYMBOL)
2826 {
2827 text = GetAttributes().GetBulletSymbol();
2828 }
2829
2830 if (GetAttributes().GetBulletStyle() & wxTEXT_ATTR_BULLET_STYLE_PARENTHESES)
2831 {
2832 text = wxT("(") + text + wxT(")");
2833 }
2834 if (GetAttributes().GetBulletStyle() & wxTEXT_ATTR_BULLET_STYLE_PERIOD)
2835 {
2836 text += wxT(".");
2837 }
2838
2839 return text;
2840}
2841
1e967276
JS
2842/// Allocate or reuse a line object
2843wxRichTextLine* wxRichTextParagraph::AllocateLine(int pos)
2844{
2845 if (pos < (int) m_cachedLines.GetCount())
2846 {
2847 wxRichTextLine* line = m_cachedLines.Item(pos)->GetData();
2848 line->Init(this);
2849 return line;
2850 }
2851 else
2852 {
2853 wxRichTextLine* line = new wxRichTextLine(this);
2854 m_cachedLines.Append(line);
2855 return line;
2856 }
2857}
2858
2859/// Clear remaining unused line objects, if any
2860bool wxRichTextParagraph::ClearUnusedLines(int lineCount)
2861{
2862 int cachedLineCount = m_cachedLines.GetCount();
2863 if ((int) cachedLineCount > lineCount)
2864 {
2865 for (int i = 0; i < (int) (cachedLineCount - lineCount); i ++)
2866 {
2867 wxRichTextLineList::compatibility_iterator node = m_cachedLines.GetLast();
2868 wxRichTextLine* line = node->GetData();
2869 m_cachedLines.Erase(node);
2870 delete line;
2871 }
2872 }
2873 return true;
2874}
2875
5d7836c4
JS
2876
2877/*!
2878 * wxRichTextLine
2879 * This object represents a line in a paragraph, and stores
2880 * offsets from the start of the paragraph representing the
2881 * start and end positions of the line.
2882 */
2883
2884wxRichTextLine::wxRichTextLine(wxRichTextParagraph* parent)
2885{
1e967276 2886 Init(parent);
5d7836c4
JS
2887}
2888
2889/// Initialisation
1e967276 2890void wxRichTextLine::Init(wxRichTextParagraph* parent)
5d7836c4 2891{
1e967276
JS
2892 m_parent = parent;
2893 m_range.SetRange(-1, -1);
2894 m_pos = wxPoint(0, 0);
2895 m_size = wxSize(0, 0);
5d7836c4
JS
2896 m_descent = 0;
2897}
2898
2899/// Copy
2900void wxRichTextLine::Copy(const wxRichTextLine& obj)
2901{
2902 m_range = obj.m_range;
2903}
2904
2905/// Get the absolute object position
2906wxPoint wxRichTextLine::GetAbsolutePosition() const
2907{
2908 return m_parent->GetPosition() + m_pos;
2909}
2910
1e967276
JS
2911/// Get the absolute range
2912wxRichTextRange wxRichTextLine::GetAbsoluteRange() const
2913{
2914 wxRichTextRange range(m_range.GetStart() + m_parent->GetRange().GetStart(), 0);
2915 range.SetEnd(range.GetStart() + m_range.GetLength()-1);
2916 return range;
2917}
2918
5d7836c4
JS
2919/*!
2920 * wxRichTextPlainText
2921 * This object represents a single piece of text.
2922 */
2923
2924IMPLEMENT_DYNAMIC_CLASS(wxRichTextPlainText, wxRichTextObject)
2925
2926wxRichTextPlainText::wxRichTextPlainText(const wxString& text, wxRichTextObject* parent, wxTextAttrEx* style):
2927 wxRichTextObject(parent)
2928{
2929 if (parent && !style)
2930 SetAttributes(parent->GetAttributes());
2931 if (style)
2932 SetAttributes(*style);
2933
2934 m_text = text;
2935}
2936
2937/// Draw the item
2938bool wxRichTextPlainText::Draw(wxDC& dc, const wxRichTextRange& range, const wxRichTextRange& selectionRange, const wxRect& rect, int descent, int WXUNUSED(style))
2939{
2940 int offset = GetRange().GetStart();
2941
2942 long len = range.GetLength();
2943 wxString stringChunk = m_text.Mid(range.GetStart() - offset, (size_t) len);
2944
2945 int charHeight = dc.GetCharHeight();
2946
2947 int x = rect.x;
2948 int y = rect.y + (rect.height - charHeight - (descent - m_descent));
2949
2950 // Test for the optimized situations where all is selected, or none
2951 // is selected.
2952
2953 if (GetAttributes().GetFont().Ok())
2954 dc.SetFont(GetAttributes().GetFont());
2955
2956 // (a) All selected.
2957 if (selectionRange.GetStart() <= range.GetStart() && selectionRange.GetEnd() >= range.GetEnd())
7f0d9d71 2958 {/*
5d7836c4
JS
2959 // Draw all selected
2960 dc.SetBrush(*wxBLACK_BRUSH);
2961 dc.SetPen(*wxBLACK_PEN);
2962 wxCoord w, h;
2963 dc.GetTextExtent(stringChunk, & w, & h);
2964 wxRect selRect(x, rect.y, w, rect.GetHeight());
2965 dc.DrawRectangle(selRect);
2966 dc.SetTextForeground(*wxWHITE);
2967 dc.SetBackgroundMode(wxTRANSPARENT);
7f0d9d71
JS
2968 dc.DrawText(stringChunk, x, y);*/
2969 DrawTabbedString(dc, rect,stringChunk, x, y, true);
5d7836c4
JS
2970 }
2971 // (b) None selected.
2972 else if (selectionRange.GetEnd() < range.GetStart() || selectionRange.GetStart() > range.GetEnd())
2973 {
2974 // Draw all unselected
7f0d9d71 2975 /*
5d7836c4
JS
2976 dc.SetTextForeground(GetAttributes().GetTextColour());
2977 dc.SetBackgroundMode(wxTRANSPARENT);
7f0d9d71
JS
2978 dc.DrawText(stringChunk, x, y);*/
2979 DrawTabbedString(dc, rect,stringChunk, x, y, false);
5d7836c4
JS
2980 }
2981 else
2982 {
2983 // (c) Part selected, part not
2984 // Let's draw unselected chunk, selected chunk, then unselected chunk.
2985
2986 dc.SetBackgroundMode(wxTRANSPARENT);
7fe8059f 2987
5d7836c4
JS
2988 // 1. Initial unselected chunk, if any, up until start of selection.
2989 if (selectionRange.GetStart() > range.GetStart() && selectionRange.GetStart() <= range.GetEnd())
2990 {
2991 int r1 = range.GetStart();
2992 int s1 = selectionRange.GetStart()-1;
2993 int fragmentLen = s1 - r1 + 1;
2994 if (fragmentLen < 0)
2995 wxLogDebug(wxT("Mid(%d, %d"), (int)(r1 - offset), (int)fragmentLen);
2996 wxString stringFragment = m_text.Mid(r1 - offset, fragmentLen);
7f0d9d71 2997/*
5d7836c4
JS
2998 dc.SetTextForeground(GetAttributes().GetTextColour());
2999 dc.DrawText(stringFragment, x, y);
3000
3001 wxCoord w, h;
3002 dc.GetTextExtent(stringFragment, & w, & h);
7f0d9d71
JS
3003 x += w;*/
3004 DrawTabbedString(dc, rect,stringFragment, x, y, false);
5d7836c4
JS
3005 }
3006
3007 // 2. Selected chunk, if any.
3008 if (selectionRange.GetEnd() >= range.GetStart())
3009 {
3010 int s1 = wxMax(selectionRange.GetStart(), range.GetStart());
3011 int s2 = wxMin(selectionRange.GetEnd(), range.GetEnd());
3012
3013 int fragmentLen = s2 - s1 + 1;
3014 if (fragmentLen < 0)
3015 wxLogDebug(wxT("Mid(%d, %d"), (int)(s1 - offset), (int)fragmentLen);
3016 wxString stringFragment = m_text.Mid(s1 - offset, fragmentLen);
7f0d9d71 3017/*
5d7836c4
JS
3018 wxCoord w, h;
3019 dc.GetTextExtent(stringFragment, & w, & h);
3020 wxRect selRect(x, rect.y, w, rect.GetHeight());
3021
3022 dc.SetBrush(*wxBLACK_BRUSH);
3023 dc.SetPen(*wxBLACK_PEN);
3024 dc.DrawRectangle(selRect);
3025 dc.SetTextForeground(*wxWHITE);
3026 dc.DrawText(stringFragment, x, y);
3027
7f0d9d71
JS
3028 x += w;*/
3029 DrawTabbedString(dc, rect,stringFragment, x, y, true);
5d7836c4
JS
3030 }
3031
3032 // 3. Remaining unselected chunk, if any
3033 if (selectionRange.GetEnd() < range.GetEnd())
3034 {
3035 int s2 = wxMin(selectionRange.GetEnd()+1, range.GetEnd());
3036 int r2 = range.GetEnd();
3037
3038 int fragmentLen = r2 - s2 + 1;
3039 if (fragmentLen < 0)
3040 wxLogDebug(wxT("Mid(%d, %d"), (int)(s2 - offset), (int)fragmentLen);
3041 wxString stringFragment = m_text.Mid(s2 - offset, fragmentLen);
7f0d9d71 3042/*
5d7836c4 3043 dc.SetTextForeground(GetAttributes().GetTextColour());
7f0d9d71
JS
3044 dc.DrawText(stringFragment, x, y);*/
3045 DrawTabbedString(dc, rect,stringFragment, x, y, false);
7fe8059f 3046 }
5d7836c4
JS
3047 }
3048
3049 return true;
3050}
61399247 3051
7f0d9d71
JS
3052bool wxRichTextPlainText::DrawTabbedString(wxDC& dc,const wxRect& rect,wxString& str, wxCoord& x, wxCoord& y, bool selected)
3053{
3054 wxArrayInt tab_array = GetAttributes().GetTabs();
3055 if(tab_array.IsEmpty()){// create a default tab list at 10 mm each.
3056 for( int i = 0; i < 20; ++i){
3057 tab_array.Add(i*100);
3058 }
3059 }
3060 int map_mode = dc.GetMapMode();
3061 dc.SetMapMode(wxMM_LOMETRIC );
3062 int num_tabs = tab_array.GetCount();
3063 for( int i = 0; i < num_tabs; ++i){
61399247 3064 tab_array[i] = dc.LogicalToDeviceXRel(tab_array[i]);
7f0d9d71
JS
3065 }
3066 dc.SetMapMode(map_mode );
3067 int next_tab_pos = -1;
3068 int tab_pos = -1;
3069 wxCoord w, h;
3070 if(selected){
3071 dc.SetBrush(*wxBLACK_BRUSH);
3072 dc.SetPen(*wxBLACK_PEN);
3073 dc.SetTextForeground(*wxWHITE);
3074 dc.SetBackgroundMode(wxTRANSPARENT);
3075 }
3076 else{
3077 dc.SetTextForeground(GetAttributes().GetTextColour());
3078 dc.SetBackgroundMode(wxTRANSPARENT);
3079 }
61399247 3080 while(str.Find(wxT('\t')) >= 0){// the string has a tab
7f0d9d71
JS
3081 // break up the string at the Tab
3082 wxString stringChunk = str.BeforeFirst(wxT('\t'));
3083 str = str.AfterFirst(wxT('\t'));
3084 dc.GetTextExtent(stringChunk, & w, & h);
3085 tab_pos = x + w;
3086 bool not_found = true;
3087 for( int i = 0; i < num_tabs && not_found; ++i){
3088 next_tab_pos = tab_array.Item(i);
3089 if( next_tab_pos > tab_pos){
3090 not_found = false;
3091 if(selected){
3092 w = next_tab_pos - x;
3093 wxRect selRect(x, rect.y, w, rect.GetHeight());
61399247 3094 dc.DrawRectangle(selRect);
7f0d9d71
JS
3095 }
3096 dc.DrawText(stringChunk, x, y);
3097 x = next_tab_pos;
3098 }
3099 }
3100 }
61399247 3101
7f0d9d71
JS
3102 dc.GetTextExtent(str, & w, & h);
3103 if(selected){
3104 wxRect selRect(x, rect.y, w, rect.GetHeight());
61399247 3105 dc.DrawRectangle(selRect);
7f0d9d71
JS
3106 }
3107 dc.DrawText(str, x, y);
3108 x += w;
3109 return true;
5d7836c4 3110
7f0d9d71 3111}
5d7836c4 3112/// Lay the item out
38113684 3113bool wxRichTextPlainText::Layout(wxDC& dc, const wxRect& WXUNUSED(rect), int WXUNUSED(style))
5d7836c4
JS
3114{
3115 if (GetAttributes().GetFont().Ok())
3116 dc.SetFont(GetAttributes().GetFont());
3117
3118 wxCoord w, h;
3119 dc.GetTextExtent(m_text, & w, & h, & m_descent);
3120 m_size = wxSize(w, dc.GetCharHeight());
3121
3122 return true;
3123}
3124
3125/// Copy
3126void wxRichTextPlainText::Copy(const wxRichTextPlainText& obj)
3127{
3128 wxRichTextObject::Copy(obj);
3129
3130 m_text = obj.m_text;
3131}
3132
3133/// Get/set the object size for the given range. Returns false if the range
3134/// is invalid for this object.
7f0d9d71 3135bool wxRichTextPlainText::GetRangeSize(const wxRichTextRange& range, wxSize& size, int& descent, wxDC& dc, int WXUNUSED(flags), wxPoint position) const
5d7836c4
JS
3136{
3137 if (!range.IsWithin(GetRange()))
3138 return false;
3139
3140 // Always assume unformatted text, since at this level we have no knowledge
3141 // of line breaks - and we don't need it, since we'll calculate size within
3142 // formatted text by doing it in chunks according to the line ranges
3143
3144 if (GetAttributes().GetFont().Ok())
3145 dc.SetFont(GetAttributes().GetFont());
3146
3147 int startPos = range.GetStart() - GetRange().GetStart();
3148 long len = range.GetLength();
3149 wxString stringChunk = m_text.Mid(startPos, (size_t) len);
3150 wxCoord w, h;
7f0d9d71 3151 int width = 0;
61399247 3152 if(stringChunk.Find(wxT('\t')) >= 0){// the string has a tab
7f0d9d71 3153 wxArrayInt tab_array = GetAttributes().GetTabs();
61399247
WS
3154 if(tab_array.IsEmpty())
3155 {
3156 // create a default tab list at 10 mm each.
3157 for( int i = 0; i < 20; ++i)
3158 {
7f0d9d71
JS
3159 tab_array.Add(i*100);
3160 }
3161 }
3162 int map_mode = dc.GetMapMode();
3163 dc.SetMapMode(wxMM_LOMETRIC );
3164 int num_tabs = tab_array.GetCount();
61399247
WS
3165 for( int i = 0; i < num_tabs; ++i)
3166 {
3167 tab_array[i] = dc.LogicalToDeviceXRel(tab_array[i]);
7f0d9d71
JS
3168 }
3169 dc.SetMapMode(map_mode );
3170 int next_tab_pos = -1;
61399247
WS
3171
3172 while(stringChunk.Find(wxT('\t')) >= 0){// the string has a tab
7f0d9d71
JS
3173 // break up the string at the Tab
3174 wxString stringFragment = stringChunk.BeforeFirst(wxT('\t'));
3175 stringChunk = stringChunk.AfterFirst(wxT('\t'));
3176 dc.GetTextExtent(stringFragment, & w, & h);
3177 width += w;
3178 int absolute_width = width + position.x;
3179 bool not_found = true;
3180 for( int i = 0; i < num_tabs && not_found; ++i){
3181 next_tab_pos = tab_array.Item(i);
3182 if( next_tab_pos > absolute_width){
61399247 3183 not_found = false;
7f0d9d71
JS
3184 width = next_tab_pos - position.x;
3185 }
3186 }
3187 }
3188 }
5d7836c4 3189 dc.GetTextExtent(stringChunk, & w, & h, & descent);
7f0d9d71
JS
3190 width += w;
3191 size = wxSize(width, dc.GetCharHeight());
61399247 3192
5d7836c4
JS
3193 return true;
3194}
3195
3196/// Do a split, returning an object containing the second part, and setting
3197/// the first part in 'this'.
3198wxRichTextObject* wxRichTextPlainText::DoSplit(long pos)
3199{
3200 int index = pos - GetRange().GetStart();
28f92d74 3201 if (index < 0 || index >= (int) m_text.length())
5d7836c4
JS
3202 return NULL;
3203
3204 wxString firstPart = m_text.Mid(0, index);
3205 wxString secondPart = m_text.Mid(index);
3206
3207 m_text = firstPart;
3208
3209 wxRichTextPlainText* newObject = new wxRichTextPlainText(secondPart);
3210 newObject->SetAttributes(GetAttributes());
3211
3212 newObject->SetRange(wxRichTextRange(pos, GetRange().GetEnd()));
3213 GetRange().SetEnd(pos-1);
3214
3215 return newObject;
3216}
3217
3218/// Calculate range
3219void wxRichTextPlainText::CalculateRange(long start, long& end)
3220{
28f92d74 3221 end = start + m_text.length() - 1;
5d7836c4
JS
3222 m_range.SetRange(start, end);
3223}
3224
3225/// Delete range
3226bool wxRichTextPlainText::DeleteRange(const wxRichTextRange& range)
3227{
3228 wxRichTextRange r = range;
3229
3230 r.LimitTo(GetRange());
3231
3232 if (r.GetStart() == GetRange().GetStart() && r.GetEnd() == GetRange().GetEnd())
3233 {
3234 m_text.Empty();
3235 return true;
3236 }
3237
3238 long startIndex = r.GetStart() - GetRange().GetStart();
3239 long len = r.GetLength();
3240
3241 m_text = m_text.Mid(0, startIndex) + m_text.Mid(startIndex+len);
3242 return true;
3243}
3244
3245/// Get text for the given range.
3246wxString wxRichTextPlainText::GetTextForRange(const wxRichTextRange& range) const
3247{
3248 wxRichTextRange r = range;
3249
3250 r.LimitTo(GetRange());
3251
3252 long startIndex = r.GetStart() - GetRange().GetStart();
3253 long len = r.GetLength();
3254
3255 return m_text.Mid(startIndex, len);
3256}
3257
3258/// Returns true if this object can merge itself with the given one.
3259bool wxRichTextPlainText::CanMerge(wxRichTextObject* object) const
3260{
3261 return object->GetClassInfo() == CLASSINFO(wxRichTextPlainText) &&
7fe8059f 3262 (m_text.empty() || wxTextAttrEq(GetAttributes(), object->GetAttributes()));
5d7836c4
JS
3263}
3264
3265/// Returns true if this object merged itself with the given one.
3266/// The calling code will then delete the given object.
3267bool wxRichTextPlainText::Merge(wxRichTextObject* object)
3268{
3269 wxRichTextPlainText* textObject = wxDynamicCast(object, wxRichTextPlainText);
3270 wxASSERT( textObject != NULL );
3271
3272 if (textObject)
3273 {
3274 m_text += textObject->GetText();
3275 return true;
3276 }
3277 else
3278 return false;
3279}
3280
3281/// Dump to output stream for debugging
3282void wxRichTextPlainText::Dump(wxTextOutputStream& stream)
3283{
3284 wxRichTextObject::Dump(stream);
3285 stream << m_text << wxT("\n");
3286}
3287
3288/*!
3289 * wxRichTextBuffer
3290 * This is a kind of box, used to represent the whole buffer
3291 */
3292
3293IMPLEMENT_DYNAMIC_CLASS(wxRichTextBuffer, wxRichTextParagraphLayoutBox)
3294
3295wxList wxRichTextBuffer::sm_handlers;
3296
3297/// Initialisation
3298void wxRichTextBuffer::Init()
3299{
3300 m_commandProcessor = new wxCommandProcessor;
3301 m_styleSheet = NULL;
3302 m_modified = false;
3303 m_batchedCommandDepth = 0;
3304 m_batchedCommand = NULL;
3305 m_suppressUndo = 0;
3306}
3307
3308/// Initialisation
3309wxRichTextBuffer::~wxRichTextBuffer()
3310{
3311 delete m_commandProcessor;
3312 delete m_batchedCommand;
3313
3314 ClearStyleStack();
3315}
3316
3317void wxRichTextBuffer::Clear()
3318{
3319 DeleteChildren();
3320 GetCommandProcessor()->ClearCommands();
3321 Modify(false);
1e967276 3322 Invalidate(wxRICHTEXT_ALL);
5d7836c4
JS
3323}
3324
3325void wxRichTextBuffer::Reset()
3326{
3327 DeleteChildren();
7fe8059f 3328 AddParagraph(wxEmptyString);
5d7836c4
JS
3329 GetCommandProcessor()->ClearCommands();
3330 Modify(false);
1e967276 3331 Invalidate(wxRICHTEXT_ALL);
5d7836c4
JS
3332}
3333
3334/// Submit command to insert the given text
3335bool wxRichTextBuffer::InsertTextWithUndo(long pos, const wxString& text, wxRichTextCtrl* ctrl)
3336{
3337 wxRichTextAction* action = new wxRichTextAction(NULL, _("Insert Text"), wxRICHTEXT_INSERT, this, ctrl, false);
3338
3339 action->GetNewParagraphs().AddParagraphs(text);
3340 if (action->GetNewParagraphs().GetChildCount() == 1)
3341 action->GetNewParagraphs().SetPartialParagraph(true);
3342
3343 action->SetPosition(pos);
3344
3345 // Set the range we'll need to delete in Undo
28f92d74 3346 action->SetRange(wxRichTextRange(pos, pos + text.length() - 1));
7fe8059f 3347
5d7836c4 3348 SubmitAction(action);
7fe8059f 3349
5d7836c4
JS
3350 return true;
3351}
3352
3353/// Submit command to insert the given text
3354bool wxRichTextBuffer::InsertNewlineWithUndo(long pos, wxRichTextCtrl* ctrl)
3355{
3356 wxRichTextAction* action = new wxRichTextAction(NULL, _("Insert Text"), wxRICHTEXT_INSERT, this, ctrl, false);
3357
3358 wxTextAttrEx attr(GetBasicStyle());
3359 wxRichTextApplyStyle(attr, GetDefaultStyle());
7fe8059f
WS
3360
3361 wxRichTextParagraph* newPara = new wxRichTextParagraph(wxEmptyString, this, & attr);
5d7836c4
JS
3362 action->GetNewParagraphs().AppendChild(newPara);
3363 action->GetNewParagraphs().UpdateRanges();
3364 action->GetNewParagraphs().SetPartialParagraph(false);
3365 action->SetPosition(pos);
3366
3367 // Set the range we'll need to delete in Undo
3368 action->SetRange(wxRichTextRange(pos, pos));
7fe8059f 3369
5d7836c4 3370 SubmitAction(action);
7fe8059f 3371
5d7836c4
JS
3372 return true;
3373}
3374
3375/// Submit command to insert the given image
3376bool wxRichTextBuffer::InsertImageWithUndo(long pos, const wxRichTextImageBlock& imageBlock, wxRichTextCtrl* ctrl)
3377{
3378 wxRichTextAction* action = new wxRichTextAction(NULL, _("Insert Image"), wxRICHTEXT_INSERT, this, ctrl, false);
3379
3380 wxTextAttrEx attr(GetBasicStyle());
3381 wxRichTextApplyStyle(attr, GetDefaultStyle());
7fe8059f 3382
5d7836c4
JS
3383 wxRichTextParagraph* newPara = new wxRichTextParagraph(this, & attr);
3384 wxRichTextImage* imageObject = new wxRichTextImage(imageBlock, newPara);
3385 newPara->AppendChild(imageObject);
3386 action->GetNewParagraphs().AppendChild(newPara);
3387 action->GetNewParagraphs().UpdateRanges();
3388
3389 action->GetNewParagraphs().SetPartialParagraph(true);
3390
3391 action->SetPosition(pos);
3392
3393 // Set the range we'll need to delete in Undo
3394 action->SetRange(wxRichTextRange(pos, pos));
7fe8059f 3395
5d7836c4 3396 SubmitAction(action);
7fe8059f 3397
5d7836c4
JS
3398 return true;
3399}
3400
3401/// Submit command to delete this range
3402bool wxRichTextBuffer::DeleteRangeWithUndo(const wxRichTextRange& range, long initialCaretPosition, long WXUNUSED(newCaretPositon), wxRichTextCtrl* ctrl)
3403{
3404 wxRichTextAction* action = new wxRichTextAction(NULL, _("Delete"), wxRICHTEXT_DELETE, this, ctrl);
7fe8059f 3405
5d7836c4
JS
3406 action->SetPosition(initialCaretPosition);
3407
3408 // Set the range to delete
3409 action->SetRange(range);
7fe8059f 3410
5d7836c4
JS
3411 // Copy the fragment that we'll need to restore in Undo
3412 CopyFragment(range, action->GetOldParagraphs());
3413
3414 // Special case: if there is only one (non-partial) paragraph,
3415 // we must save the *next* paragraph's style, because that
3416 // is the style we must apply when inserting the content back
3417 // when undoing the delete. (This is because we're merging the
3418 // paragraph with the previous paragraph and throwing away
3419 // the style, and we need to restore it.)
3420 if (!action->GetOldParagraphs().GetPartialParagraph() && action->GetOldParagraphs().GetChildCount() == 1)
3421 {
3422 wxRichTextParagraph* lastPara = GetParagraphAtPosition(range.GetStart());
3423 if (lastPara)
3424 {
3425 wxRichTextParagraph* nextPara = GetParagraphAtPosition(range.GetEnd()+1);
3426 if (nextPara)
3427 {
3428 wxRichTextParagraph* para = (wxRichTextParagraph*) action->GetOldParagraphs().GetChild(0);
3429 para->SetAttributes(nextPara->GetAttributes());
3430 }
3431 }
3432 }
3433
3434 SubmitAction(action);
7fe8059f 3435
5d7836c4
JS
3436 return true;
3437}
3438
3439/// Collapse undo/redo commands
3440bool wxRichTextBuffer::BeginBatchUndo(const wxString& cmdName)
3441{
3442 if (m_batchedCommandDepth == 0)
3443 {
3444 wxASSERT(m_batchedCommand == NULL);
3445 if (m_batchedCommand)
3446 {
3447 GetCommandProcessor()->Submit(m_batchedCommand);
3448 }
3449 m_batchedCommand = new wxRichTextCommand(cmdName);
3450 }
3451
7fe8059f 3452 m_batchedCommandDepth ++;
5d7836c4
JS
3453
3454 return true;
3455}
3456
3457/// Collapse undo/redo commands
3458bool wxRichTextBuffer::EndBatchUndo()
3459{
3460 m_batchedCommandDepth --;
3461
3462 wxASSERT(m_batchedCommandDepth >= 0);
3463 wxASSERT(m_batchedCommand != NULL);
3464
3465 if (m_batchedCommandDepth == 0)
3466 {
3467 GetCommandProcessor()->Submit(m_batchedCommand);
3468 m_batchedCommand = NULL;
3469 }
3470
3471 return true;
3472}
3473
3474/// Submit immediately, or delay according to whether collapsing is on
3475bool wxRichTextBuffer::SubmitAction(wxRichTextAction* action)
3476{
3477 if (BatchingUndo() && m_batchedCommand && !SuppressingUndo())
3478 m_batchedCommand->AddAction(action);
3479 else
3480 {
3481 wxRichTextCommand* cmd = new wxRichTextCommand(action->GetName());
3482 cmd->AddAction(action);
3483
3484 // Only store it if we're not suppressing undo.
3485 return GetCommandProcessor()->Submit(cmd, !SuppressingUndo());
3486 }
3487
3488 return true;
3489}
3490
3491/// Begin suppressing undo/redo commands.
3492bool wxRichTextBuffer::BeginSuppressUndo()
3493{
7fe8059f 3494 m_suppressUndo ++;
5d7836c4
JS
3495
3496 return true;
3497}
3498
3499/// End suppressing undo/redo commands.
3500bool wxRichTextBuffer::EndSuppressUndo()
3501{
7fe8059f 3502 m_suppressUndo --;
5d7836c4
JS
3503
3504 return true;
3505}
3506
3507/// Begin using a style
3508bool wxRichTextBuffer::BeginStyle(const wxTextAttrEx& style)
3509{
3510 wxTextAttrEx newStyle(GetDefaultStyle());
3511
3512 // Save the old default style
3513 m_attributeStack.Append((wxObject*) new wxTextAttrEx(GetDefaultStyle()));
3514
3515 wxRichTextApplyStyle(newStyle, style);
3516 newStyle.SetFlags(style.GetFlags()|newStyle.GetFlags());
3517
3518 SetDefaultStyle(newStyle);
3519
3520 // wxLogDebug("Default style size = %d", GetDefaultStyle().GetFont().GetPointSize());
3521
3522 return true;
3523}
3524
3525/// End the style
3526bool wxRichTextBuffer::EndStyle()
3527{
63886f6d 3528 if (!m_attributeStack.GetFirst())
5d7836c4
JS
3529 {
3530 wxLogDebug(_("Too many EndStyle calls!"));
3531 return false;
3532 }
3533
09f14108 3534 wxList::compatibility_iterator node = m_attributeStack.GetLast();
5d7836c4 3535 wxTextAttrEx* attr = (wxTextAttrEx*)node->GetData();
9e31a660 3536 m_attributeStack.Erase(node);
5d7836c4
JS
3537
3538 SetDefaultStyle(*attr);
3539
3540 delete attr;
3541 return true;
3542}
3543
3544/// End all styles
3545bool wxRichTextBuffer::EndAllStyles()
3546{
3547 while (m_attributeStack.GetCount() != 0)
3548 EndStyle();
3549 return true;
3550}
3551
3552/// Clear the style stack
3553void wxRichTextBuffer::ClearStyleStack()
3554{
09f14108 3555 for (wxList::compatibility_iterator node = m_attributeStack.GetFirst(); node; node = node->GetNext())
5d7836c4
JS
3556 delete (wxTextAttrEx*) node->GetData();
3557 m_attributeStack.Clear();
3558}
3559
3560/// Begin using bold
3561bool wxRichTextBuffer::BeginBold()
3562{
3563 wxFont font(GetBasicStyle().GetFont());
3564 font.SetWeight(wxBOLD);
3565
3566 wxTextAttrEx attr;
3567 attr.SetFont(font,wxTEXT_ATTR_FONT_WEIGHT);
7fe8059f 3568
5d7836c4
JS
3569 return BeginStyle(attr);
3570}
3571
3572/// Begin using italic
3573bool wxRichTextBuffer::BeginItalic()
3574{
3575 wxFont font(GetBasicStyle().GetFont());
3576 font.SetStyle(wxITALIC);
3577
3578 wxTextAttrEx attr;
3579 attr.SetFont(font, wxTEXT_ATTR_FONT_ITALIC);
7fe8059f 3580
5d7836c4
JS
3581 return BeginStyle(attr);
3582}
3583
3584/// Begin using underline
3585bool wxRichTextBuffer::BeginUnderline()
3586{
3587 wxFont font(GetBasicStyle().GetFont());
3588 font.SetUnderlined(true);
3589
3590 wxTextAttrEx attr;
3591 attr.SetFont(font, wxTEXT_ATTR_FONT_UNDERLINE);
7fe8059f 3592
5d7836c4
JS
3593 return BeginStyle(attr);
3594}
3595
3596/// Begin using point size
3597bool wxRichTextBuffer::BeginFontSize(int pointSize)
3598{
3599 wxFont font(GetBasicStyle().GetFont());
3600 font.SetPointSize(pointSize);
3601
3602 wxTextAttrEx attr;
3603 attr.SetFont(font, wxTEXT_ATTR_FONT_SIZE);
7fe8059f 3604
5d7836c4
JS
3605 return BeginStyle(attr);
3606}
3607
3608/// Begin using this font
3609bool wxRichTextBuffer::BeginFont(const wxFont& font)
3610{
3611 wxTextAttrEx attr;
3612 attr.SetFlags(wxTEXT_ATTR_FONT);
3613 attr.SetFont(font);
7fe8059f 3614
5d7836c4
JS
3615 return BeginStyle(attr);
3616}
3617
3618/// Begin using this colour
3619bool wxRichTextBuffer::BeginTextColour(const wxColour& colour)
3620{
3621 wxTextAttrEx attr;
3622 attr.SetFlags(wxTEXT_ATTR_TEXT_COLOUR);
3623 attr.SetTextColour(colour);
7fe8059f 3624
5d7836c4
JS
3625 return BeginStyle(attr);
3626}
3627
3628/// Begin using alignment
3629bool wxRichTextBuffer::BeginAlignment(wxTextAttrAlignment alignment)
3630{
3631 wxTextAttrEx attr;
3632 attr.SetFlags(wxTEXT_ATTR_ALIGNMENT);
3633 attr.SetAlignment(alignment);
7fe8059f 3634
5d7836c4
JS
3635 return BeginStyle(attr);
3636}
3637
3638/// Begin left indent
3639bool wxRichTextBuffer::BeginLeftIndent(int leftIndent, int leftSubIndent)
3640{
3641 wxTextAttrEx attr;
3642 attr.SetFlags(wxTEXT_ATTR_LEFT_INDENT);
3643 attr.SetLeftIndent(leftIndent, leftSubIndent);
7fe8059f 3644
5d7836c4
JS
3645 return BeginStyle(attr);
3646}
3647
3648/// Begin right indent
3649bool wxRichTextBuffer::BeginRightIndent(int rightIndent)
3650{
3651 wxTextAttrEx attr;
3652 attr.SetFlags(wxTEXT_ATTR_RIGHT_INDENT);
3653 attr.SetRightIndent(rightIndent);
7fe8059f 3654
5d7836c4
JS
3655 return BeginStyle(attr);
3656}
3657
3658/// Begin paragraph spacing
3659bool wxRichTextBuffer::BeginParagraphSpacing(int before, int after)
3660{
3661 long flags = 0;
3662 if (before != 0)
3663 flags |= wxTEXT_ATTR_PARA_SPACING_BEFORE;
3664 if (after != 0)
3665 flags |= wxTEXT_ATTR_PARA_SPACING_AFTER;
3666
3667 wxTextAttrEx attr;
3668 attr.SetFlags(flags);
3669 attr.SetParagraphSpacingBefore(before);
3670 attr.SetParagraphSpacingAfter(after);
7fe8059f 3671
5d7836c4
JS
3672 return BeginStyle(attr);
3673}
3674
3675/// Begin line spacing
3676bool wxRichTextBuffer::BeginLineSpacing(int lineSpacing)
3677{
3678 wxTextAttrEx attr;
3679 attr.SetFlags(wxTEXT_ATTR_LINE_SPACING);
3680 attr.SetLineSpacing(lineSpacing);
7fe8059f 3681
5d7836c4
JS
3682 return BeginStyle(attr);
3683}
3684
3685/// Begin numbered bullet
3686bool wxRichTextBuffer::BeginNumberedBullet(int bulletNumber, int leftIndent, int leftSubIndent, int bulletStyle)
3687{
3688 wxTextAttrEx attr;
3689 attr.SetFlags(wxTEXT_ATTR_BULLET_STYLE|wxTEXT_ATTR_BULLET_NUMBER|wxTEXT_ATTR_LEFT_INDENT);
3690 attr.SetBulletStyle(bulletStyle);
3691 attr.SetBulletNumber(bulletNumber);
3692 attr.SetLeftIndent(leftIndent, leftSubIndent);
7fe8059f 3693
5d7836c4
JS
3694 return BeginStyle(attr);
3695}
3696
3697/// Begin symbol bullet
3698bool wxRichTextBuffer::BeginSymbolBullet(wxChar symbol, int leftIndent, int leftSubIndent, int bulletStyle)
3699{
3700 wxTextAttrEx attr;
3701 attr.SetFlags(wxTEXT_ATTR_BULLET_STYLE|wxTEXT_ATTR_BULLET_SYMBOL|wxTEXT_ATTR_LEFT_INDENT);
3702 attr.SetBulletStyle(bulletStyle);
3703 attr.SetLeftIndent(leftIndent, leftSubIndent);
3704 attr.SetBulletSymbol(symbol);
7fe8059f 3705
5d7836c4
JS
3706 return BeginStyle(attr);
3707}
3708
3709/// Begin named character style
3710bool wxRichTextBuffer::BeginCharacterStyle(const wxString& characterStyle)
3711{
3712 if (GetStyleSheet())
3713 {
3714 wxRichTextCharacterStyleDefinition* def = GetStyleSheet()->FindCharacterStyle(characterStyle);
3715 if (def)
3716 {
3717 wxTextAttrEx attr;
3718 def->GetStyle().CopyTo(attr);
3719 return BeginStyle(attr);
3720 }
3721 }
3722 return false;
3723}
3724
3725/// Begin named paragraph style
3726bool wxRichTextBuffer::BeginParagraphStyle(const wxString& paragraphStyle)
3727{
3728 if (GetStyleSheet())
3729 {
3730 wxRichTextParagraphStyleDefinition* def = GetStyleSheet()->FindParagraphStyle(paragraphStyle);
3731 if (def)
3732 {
3733 wxTextAttrEx attr;
3734 def->GetStyle().CopyTo(attr);
3735 return BeginStyle(attr);
3736 }
3737 }
3738 return false;
3739}
3740
3741/// Adds a handler to the end
3742void wxRichTextBuffer::AddHandler(wxRichTextFileHandler *handler)
3743{
3744 sm_handlers.Append(handler);
3745}
3746
3747/// Inserts a handler at the front
3748void wxRichTextBuffer::InsertHandler(wxRichTextFileHandler *handler)
3749{
3750 sm_handlers.Insert( handler );
3751}
3752
3753/// Removes a handler
3754bool wxRichTextBuffer::RemoveHandler(const wxString& name)
3755{
3756 wxRichTextFileHandler *handler = FindHandler(name);
3757 if (handler)
3758 {
3759 sm_handlers.DeleteObject(handler);
3760 delete handler;
3761 return true;
3762 }
3763 else
3764 return false;
3765}
3766
3767/// Finds a handler by filename or, if supplied, type
3768wxRichTextFileHandler *wxRichTextBuffer::FindHandlerFilenameOrType(const wxString& filename, int imageType)
3769{
3770 if (imageType != wxRICHTEXT_TYPE_ANY)
3771 return FindHandler(imageType);
3772 else
3773 {
3774 wxString path, file, ext;
3775 wxSplitPath(filename, & path, & file, & ext);
3776 return FindHandler(ext, imageType);
3777 }
3778}
3779
3780
3781/// Finds a handler by name
3782wxRichTextFileHandler* wxRichTextBuffer::FindHandler(const wxString& name)
3783{
3784 wxList::compatibility_iterator node = sm_handlers.GetFirst();
3785 while (node)
3786 {
3787 wxRichTextFileHandler *handler = (wxRichTextFileHandler*)node->GetData();
3788 if (handler->GetName().Lower() == name.Lower()) return handler;
3789
3790 node = node->GetNext();
3791 }
3792 return NULL;
3793}
3794
3795/// Finds a handler by extension and type
3796wxRichTextFileHandler* wxRichTextBuffer::FindHandler(const wxString& extension, int type)
3797{
3798 wxList::compatibility_iterator node = sm_handlers.GetFirst();
3799 while (node)
3800 {
3801 wxRichTextFileHandler *handler = (wxRichTextFileHandler*)node->GetData();
3802 if ( handler->GetExtension().Lower() == extension.Lower() &&
3803 (type == wxRICHTEXT_TYPE_ANY || handler->GetType() == type) )
3804 return handler;
3805 node = node->GetNext();
3806 }
3807 return 0;
3808}
3809
3810/// Finds a handler by type
3811wxRichTextFileHandler* wxRichTextBuffer::FindHandler(int type)
3812{
3813 wxList::compatibility_iterator node = sm_handlers.GetFirst();
3814 while (node)
3815 {
3816 wxRichTextFileHandler *handler = (wxRichTextFileHandler *)node->GetData();
3817 if (handler->GetType() == type) return handler;
3818 node = node->GetNext();
3819 }
3820 return NULL;
3821}
3822
3823void wxRichTextBuffer::InitStandardHandlers()
3824{
3825 if (!FindHandler(wxRICHTEXT_TYPE_TEXT))
3826 AddHandler(new wxRichTextPlainTextHandler);
3827}
3828
3829void wxRichTextBuffer::CleanUpHandlers()
3830{
3831 wxList::compatibility_iterator node = sm_handlers.GetFirst();
3832 while (node)
3833 {
3834 wxRichTextFileHandler* handler = (wxRichTextFileHandler*)node->GetData();
3835 wxList::compatibility_iterator next = node->GetNext();
3836 delete handler;
3837 node = next;
3838 }
3839
3840 sm_handlers.Clear();
3841}
3842
1e967276 3843wxString wxRichTextBuffer::GetExtWildcard(bool combine, bool save, wxArrayInt* types)
5d7836c4 3844{
1e967276
JS
3845 if (types)
3846 types->Clear();
3847
5d7836c4
JS
3848 wxString wildcard;
3849
3850 wxList::compatibility_iterator node = GetHandlers().GetFirst();
3851 int count = 0;
3852 while (node)
3853 {
3854 wxRichTextFileHandler* handler = (wxRichTextFileHandler*) node->GetData();
3855 if (handler->IsVisible() && ((save && handler->CanSave()) || !save && handler->CanLoad()))
3856 {
3857 if (combine)
3858 {
3859 if (count > 0)
3860 wildcard += wxT(";");
3861 wildcard += wxT("*.") + handler->GetExtension();
3862 }
3863 else
3864 {
3865 if (count > 0)
3866 wildcard += wxT("|");
3867 wildcard += handler->GetName();
3868 wildcard += wxT(" ");
3869 wildcard += _("files");
3870 wildcard += wxT(" (*.");
3871 wildcard += handler->GetExtension();
3872 wildcard += wxT(")|*.");
3873 wildcard += handler->GetExtension();
1e967276
JS
3874 if (types)
3875 types->Add(handler->GetType());
5d7836c4
JS
3876 }
3877 count ++;
3878 }
3879
3880 node = node->GetNext();
3881 }
3882
3883 if (combine)
3884 wildcard = wxT("(") + wildcard + wxT(")|") + wildcard;
3885 return wildcard;
3886}
3887
3888/// Load a file
3889bool wxRichTextBuffer::LoadFile(const wxString& filename, int type)
3890{
3891 wxRichTextFileHandler* handler = FindHandlerFilenameOrType(filename, type);
3892 if (handler)
1e967276
JS
3893 {
3894 SetDefaultStyle(wxTextAttrEx());
3895
3896 bool success = handler->LoadFile(this, filename);
3897 Invalidate(wxRICHTEXT_ALL);
3898 return success;
3899 }
5d7836c4
JS
3900 else
3901 return false;
3902}
3903
3904/// Save a file
3905bool wxRichTextBuffer::SaveFile(const wxString& filename, int type)
3906{
3907 wxRichTextFileHandler* handler = FindHandlerFilenameOrType(filename, type);
3908 if (handler)
3909 return handler->SaveFile(this, filename);
3910 else
3911 return false;
3912}
3913
3914/// Load from a stream
3915bool wxRichTextBuffer::LoadFile(wxInputStream& stream, int type)
3916{
3917 wxRichTextFileHandler* handler = FindHandler(type);
3918 if (handler)
1e967276
JS
3919 {
3920 SetDefaultStyle(wxTextAttrEx());
3921 bool success = handler->LoadFile(this, stream);
3922 Invalidate(wxRICHTEXT_ALL);
3923 return success;
3924 }
5d7836c4
JS
3925 else
3926 return false;
3927}
3928
3929/// Save to a stream
3930bool wxRichTextBuffer::SaveFile(wxOutputStream& stream, int type)
3931{
3932 wxRichTextFileHandler* handler = FindHandler(type);
3933 if (handler)
3934 return handler->SaveFile(this, stream);
3935 else
3936 return false;
3937}
3938
3939/// Copy the range to the clipboard
3940bool wxRichTextBuffer::CopyToClipboard(const wxRichTextRange& range)
3941{
3942 bool success = false;
11ef729d 3943#if wxUSE_CLIPBOARD && wxUSE_DATAOBJ
5d7836c4 3944 wxString text = GetTextForRange(range);
d2142335 3945 if (!wxTheClipboard->IsOpened() && wxTheClipboard->Open())
7fe8059f 3946 {
5d7836c4
JS
3947 success = wxTheClipboard->SetData(new wxTextDataObject(text));
3948 wxTheClipboard->Close();
3949 }
39a1c2f2
WS
3950#else
3951 wxUnusedVar(range);
3952#endif
5d7836c4
JS
3953 return success;
3954}
3955
3956/// Paste the clipboard content to the buffer
3957bool wxRichTextBuffer::PasteFromClipboard(long position)
3958{
3959 bool success = false;
11ef729d 3960#if wxUSE_CLIPBOARD && wxUSE_DATAOBJ
5d7836c4
JS
3961 if (CanPasteFromClipboard())
3962 {
3963 if (wxTheClipboard->Open())
3964 {
3965 if (wxTheClipboard->IsSupported(wxDF_TEXT))
3966 {
3967 wxTextDataObject data;
3968 wxTheClipboard->GetData(data);
3969 wxString text(data.GetText());
d5363f57 3970 text.Replace(_T("\r\n"), _T("\n"));
5d7836c4
JS
3971
3972 InsertTextWithUndo(position+1, text, GetRichTextCtrl());
7fe8059f 3973
5d7836c4
JS
3974 success = true;
3975 }
3976 else if (wxTheClipboard->IsSupported(wxDF_BITMAP))
3977 {
3978 wxBitmapDataObject data;
3979 wxTheClipboard->GetData(data);
3980 wxBitmap bitmap(data.GetBitmap());
3981 wxImage image(bitmap.ConvertToImage());
3982
3983 wxRichTextAction* action = new wxRichTextAction(NULL, _("Insert Image"), wxRICHTEXT_INSERT, this, GetRichTextCtrl(), false);
7fe8059f 3984
5d7836c4
JS
3985 action->GetNewParagraphs().AddImage(image);
3986
3987 if (action->GetNewParagraphs().GetChildCount() == 1)
3988 action->GetNewParagraphs().SetPartialParagraph(true);
7fe8059f 3989
5d7836c4 3990 action->SetPosition(position);
7fe8059f 3991
5d7836c4
JS
3992 // Set the range we'll need to delete in Undo
3993 action->SetRange(wxRichTextRange(position, position));
7fe8059f 3994
5d7836c4
JS
3995 SubmitAction(action);
3996
3997 success = true;
3998 }
3999 wxTheClipboard->Close();
4000 }
4001 }
39a1c2f2
WS
4002#else
4003 wxUnusedVar(position);
4004#endif
5d7836c4
JS
4005 return success;
4006}
4007
4008/// Can we paste from the clipboard?
4009bool wxRichTextBuffer::CanPasteFromClipboard() const
4010{
7fe8059f 4011 bool canPaste = false;
11ef729d 4012#if wxUSE_CLIPBOARD && wxUSE_DATAOBJ
d2142335 4013 if (!wxTheClipboard->IsOpened() && wxTheClipboard->Open())
5d7836c4
JS
4014 {
4015 if (wxTheClipboard->IsSupported(wxDF_TEXT) || wxTheClipboard->IsSupported(wxDF_BITMAP))
4016 {
7fe8059f 4017 canPaste = true;
5d7836c4
JS
4018 }
4019 wxTheClipboard->Close();
4020 }
39a1c2f2 4021#endif
5d7836c4
JS
4022 return canPaste;
4023}
4024
4025/// Dumps contents of buffer for debugging purposes
4026void wxRichTextBuffer::Dump()
4027{
4028 wxString text;
4029 {
4030 wxStringOutputStream stream(& text);
4031 wxTextOutputStream textStream(stream);
4032 Dump(textStream);
4033 }
4034
4035 wxLogDebug(text);
4036}
4037
4038
4039/*
4040 * Module to initialise and clean up handlers
4041 */
4042
4043class wxRichTextModule: public wxModule
4044{
4045DECLARE_DYNAMIC_CLASS(wxRichTextModule)
4046public:
4047 wxRichTextModule() {}
4048 bool OnInit() { wxRichTextBuffer::InitStandardHandlers(); return true; };
4049 void OnExit() { wxRichTextBuffer::CleanUpHandlers(); };
4050};
4051
4052IMPLEMENT_DYNAMIC_CLASS(wxRichTextModule, wxModule)
4053
4054
4055/*!
4056 * Commands for undo/redo
4057 *
4058 */
4059
4060wxRichTextCommand::wxRichTextCommand(const wxString& name, wxRichTextCommandId id, wxRichTextBuffer* buffer,
7fe8059f 4061 wxRichTextCtrl* ctrl, bool ignoreFirstTime): wxCommand(true, name)
5d7836c4
JS
4062{
4063 /* wxRichTextAction* action = */ new wxRichTextAction(this, name, id, buffer, ctrl, ignoreFirstTime);
4064}
4065
7fe8059f 4066wxRichTextCommand::wxRichTextCommand(const wxString& name): wxCommand(true, name)
5d7836c4
JS
4067{
4068}
4069
4070wxRichTextCommand::~wxRichTextCommand()
4071{
4072 ClearActions();
4073}
4074
4075void wxRichTextCommand::AddAction(wxRichTextAction* action)
4076{
4077 if (!m_actions.Member(action))
4078 m_actions.Append(action);
4079}
4080
4081bool wxRichTextCommand::Do()
4082{
09f14108 4083 for (wxList::compatibility_iterator node = m_actions.GetFirst(); node; node = node->GetNext())
5d7836c4
JS
4084 {
4085 wxRichTextAction* action = (wxRichTextAction*) node->GetData();
4086 action->Do();
4087 }
4088
4089 return true;
4090}
4091
4092bool wxRichTextCommand::Undo()
4093{
09f14108 4094 for (wxList::compatibility_iterator node = m_actions.GetLast(); node; node = node->GetPrevious())
5d7836c4
JS
4095 {
4096 wxRichTextAction* action = (wxRichTextAction*) node->GetData();
4097 action->Undo();
4098 }
4099
4100 return true;
4101}
4102
4103void wxRichTextCommand::ClearActions()
4104{
4105 WX_CLEAR_LIST(wxList, m_actions);
4106}
4107
4108/*!
4109 * Individual action
4110 *
4111 */
4112
4113wxRichTextAction::wxRichTextAction(wxRichTextCommand* cmd, const wxString& name, wxRichTextCommandId id, wxRichTextBuffer* buffer,
4114 wxRichTextCtrl* ctrl, bool ignoreFirstTime)
4115{
4116 m_buffer = buffer;
4117 m_ignoreThis = ignoreFirstTime;
4118 m_cmdId = id;
4119 m_position = -1;
4120 m_ctrl = ctrl;
4121 m_name = name;
4122 m_newParagraphs.SetDefaultStyle(buffer->GetDefaultStyle());
4123 m_newParagraphs.SetBasicStyle(buffer->GetBasicStyle());
4124 if (cmd)
4125 cmd->AddAction(this);
4126}
4127
4128wxRichTextAction::~wxRichTextAction()
4129{
4130}
4131
4132bool wxRichTextAction::Do()
4133{
4134 m_buffer->Modify(true);
4135
4136 switch (m_cmdId)
4137 {
4138 case wxRICHTEXT_INSERT:
4139 {
4140 m_buffer->InsertFragment(GetPosition(), m_newParagraphs);
4141 m_buffer->UpdateRanges();
1e967276 4142 m_buffer->Invalidate(GetRange());
5d7836c4
JS
4143
4144 long newCaretPosition = GetPosition() + m_newParagraphs.GetRange().GetLength() - 1;
4145 if (m_newParagraphs.GetPartialParagraph())
4146 newCaretPosition --;
4147
4148 UpdateAppearance(newCaretPosition, true /* send update event */);
4149
4150 break;
4151 }
4152 case wxRICHTEXT_DELETE:
4153 {
4154 m_buffer->DeleteRange(GetRange());
4155 m_buffer->UpdateRanges();
1e967276 4156 m_buffer->Invalidate(wxRichTextRange(GetRange().GetStart(), GetRange().GetStart()));
5d7836c4
JS
4157
4158 UpdateAppearance(GetRange().GetStart()-1, true /* send update event */);
4159
4160 break;
4161 }
4162 case wxRICHTEXT_CHANGE_STYLE:
4163 {
4164 ApplyParagraphs(GetNewParagraphs());
1e967276 4165 m_buffer->Invalidate(GetRange());
5d7836c4
JS
4166
4167 UpdateAppearance(GetPosition());
4168
4169 break;
4170 }
4171 default:
4172 break;
4173 }
4174
4175 return true;
4176}
4177
4178bool wxRichTextAction::Undo()
4179{
4180 m_buffer->Modify(true);
4181
4182 switch (m_cmdId)
4183 {
4184 case wxRICHTEXT_INSERT:
4185 {
4186 m_buffer->DeleteRange(GetRange());
4187 m_buffer->UpdateRanges();
1e967276 4188 m_buffer->Invalidate(wxRichTextRange(GetRange().GetStart(), GetRange().GetStart()));
5d7836c4
JS
4189
4190 long newCaretPosition = GetPosition() - 1;
4191 // if (m_newParagraphs.GetPartialParagraph())
4192 // newCaretPosition --;
4193
4194 UpdateAppearance(newCaretPosition, true /* send update event */);
4195
4196 break;
4197 }
4198 case wxRICHTEXT_DELETE:
4199 {
4200 m_buffer->InsertFragment(GetRange().GetStart(), m_oldParagraphs);
4201 m_buffer->UpdateRanges();
1e967276 4202 m_buffer->Invalidate(GetRange());
5d7836c4
JS
4203
4204 UpdateAppearance(GetPosition(), true /* send update event */);
4205
4206 break;
4207 }
4208 case wxRICHTEXT_CHANGE_STYLE:
4209 {
4210 ApplyParagraphs(GetOldParagraphs());
1e967276 4211 m_buffer->Invalidate(GetRange());
5d7836c4
JS
4212
4213 UpdateAppearance(GetPosition());
4214
4215 break;
4216 }
4217 default:
4218 break;
4219 }
4220
4221 return true;
4222}
4223
4224/// Update the control appearance
4225void wxRichTextAction::UpdateAppearance(long caretPosition, bool sendUpdateEvent)
4226{
4227 if (m_ctrl)
4228 {
4229 m_ctrl->SetCaretPosition(caretPosition);
4230 if (!m_ctrl->IsFrozen())
4231 {
2f36e8dc 4232 m_ctrl->LayoutContent();
5d7836c4 4233 m_ctrl->PositionCaret();
76bcd815 4234 m_ctrl->Refresh(false);
5d7836c4
JS
4235
4236 if (sendUpdateEvent)
4237 m_ctrl->SendUpdateEvent();
4238 }
7fe8059f 4239 }
5d7836c4
JS
4240}
4241
4242/// Replace the buffer paragraphs with the new ones.
4243void wxRichTextAction::ApplyParagraphs(const wxRichTextFragment& fragment)
4244{
4245 wxRichTextObjectList::compatibility_iterator node = fragment.GetChildren().GetFirst();
4246 while (node)
4247 {
4248 wxRichTextParagraph* para = wxDynamicCast(node->GetData(), wxRichTextParagraph);
4249 wxASSERT (para != NULL);
4250
4251 // We'll replace the existing paragraph by finding the paragraph at this position,
4252 // delete its node data, and setting a copy as the new node data.
4253 // TODO: make more efficient by simply swapping old and new paragraph objects.
4254
4255 wxRichTextParagraph* existingPara = m_buffer->GetParagraphAtPosition(para->GetRange().GetStart());
4256 if (existingPara)
4257 {
4258 wxRichTextObjectList::compatibility_iterator bufferParaNode = m_buffer->GetChildren().Find(existingPara);
4259 if (bufferParaNode)
4260 {
4261 wxRichTextParagraph* newPara = new wxRichTextParagraph(*para);
4262 newPara->SetParent(m_buffer);
4263
4264 bufferParaNode->SetData(newPara);
4265
4266 delete existingPara;
4267 }
4268 }
4269
4270 node = node->GetNext();
4271 }
4272}
4273
4274
4275/*!
4276 * wxRichTextRange
4277 * This stores beginning and end positions for a range of data.
4278 */
4279
4280/// Limit this range to be within 'range'
4281bool wxRichTextRange::LimitTo(const wxRichTextRange& range)
4282{
4283 if (m_start < range.m_start)
4284 m_start = range.m_start;
4285
4286 if (m_end > range.m_end)
4287 m_end = range.m_end;
4288
4289 return true;
4290}
4291
4292/*!
4293 * wxRichTextImage implementation
4294 * This object represents an image.
4295 */
4296
4297IMPLEMENT_DYNAMIC_CLASS(wxRichTextImage, wxRichTextObject)
4298
4299wxRichTextImage::wxRichTextImage(const wxImage& image, wxRichTextObject* parent):
4300 wxRichTextObject(parent)
4301{
4302 m_image = image;
4303}
4304
4305wxRichTextImage::wxRichTextImage(const wxRichTextImageBlock& imageBlock, wxRichTextObject* parent):
4306 wxRichTextObject(parent)
4307{
4308 m_imageBlock = imageBlock;
4309 m_imageBlock.Load(m_image);
4310}
4311
4312/// Load wxImage from the block
4313bool wxRichTextImage::LoadFromBlock()
4314{
4315 m_imageBlock.Load(m_image);
4316 return m_imageBlock.Ok();
4317}
4318
4319/// Make block from the wxImage
4320bool wxRichTextImage::MakeBlock()
4321{
4322 if (m_imageBlock.GetImageType() == wxBITMAP_TYPE_ANY || m_imageBlock.GetImageType() == -1)
4323 m_imageBlock.SetImageType(wxBITMAP_TYPE_PNG);
4324
4325 m_imageBlock.MakeImageBlock(m_image, m_imageBlock.GetImageType());
4326 return m_imageBlock.Ok();
4327}
4328
4329
4330/// Draw the item
4331bool wxRichTextImage::Draw(wxDC& dc, const wxRichTextRange& range, const wxRichTextRange& selectionRange, const wxRect& rect, int WXUNUSED(descent), int WXUNUSED(style))
4332{
4333 if (!m_image.Ok() && m_imageBlock.Ok())
4334 LoadFromBlock();
4335
4336 if (!m_image.Ok())
4337 return false;
4338
4339 if (m_image.Ok() && !m_bitmap.Ok())
4340 m_bitmap = wxBitmap(m_image);
4341
4342 int y = rect.y + (rect.height - m_image.GetHeight());
4343
4344 if (m_bitmap.Ok())
4345 dc.DrawBitmap(m_bitmap, rect.x, y, true);
4346
4347 if (selectionRange.Contains(range.GetStart()))
4348 {
4349 dc.SetBrush(*wxBLACK_BRUSH);
4350 dc.SetPen(*wxBLACK_PEN);
4351 dc.SetLogicalFunction(wxINVERT);
4352 dc.DrawRectangle(rect);
4353 dc.SetLogicalFunction(wxCOPY);
4354 }
4355
4356 return true;
4357}
4358
4359/// Lay the item out
38113684 4360bool wxRichTextImage::Layout(wxDC& WXUNUSED(dc), const wxRect& rect, int WXUNUSED(style))
5d7836c4
JS
4361{
4362 if (!m_image.Ok())
4363 LoadFromBlock();
4364
4365 if (m_image.Ok())
4366 {
4367 SetCachedSize(wxSize(m_image.GetWidth(), m_image.GetHeight()));
4368 SetPosition(rect.GetPosition());
4369 }
4370
4371 return true;
4372}
4373
4374/// Get/set the object size for the given range. Returns false if the range
4375/// is invalid for this object.
7f0d9d71 4376bool wxRichTextImage::GetRangeSize(const wxRichTextRange& range, wxSize& size, int& WXUNUSED(descent), wxDC& WXUNUSED(dc), int WXUNUSED(flags), wxPoint WXUNUSED(position)) const
5d7836c4
JS
4377{
4378 if (!range.IsWithin(GetRange()))
4379 return false;
4380
4381 if (!m_image.Ok())
4382 return false;
4383
4384 size.x = m_image.GetWidth();
4385 size.y = m_image.GetHeight();
4386
4387 return true;
4388}
4389
4390/// Copy
4391void wxRichTextImage::Copy(const wxRichTextImage& obj)
4392{
4393 m_image = obj.m_image;
4394 m_imageBlock = obj.m_imageBlock;
4395}
4396
4397/*!
4398 * Utilities
4399 *
4400 */
4401
4402/// Compare two attribute objects
4403bool wxTextAttrEq(const wxTextAttrEx& attr1, const wxTextAttrEx& attr2)
4404{
4405 return (
4406 attr1.GetTextColour() == attr2.GetTextColour() &&
4407 attr1.GetBackgroundColour() == attr2.GetBackgroundColour() &&
4408 attr1.GetFont() == attr2.GetFont() &&
4409 attr1.GetAlignment() == attr2.GetAlignment() &&
4410 attr1.GetLeftIndent() == attr2.GetLeftIndent() &&
4411 attr1.GetRightIndent() == attr2.GetRightIndent() &&
4412 attr1.GetLeftSubIndent() == attr2.GetLeftSubIndent() &&
4413 attr1.GetTabs().GetCount() == attr2.GetTabs().GetCount() && // heuristic
4414 attr1.GetLineSpacing() == attr2.GetLineSpacing() &&
4415 attr1.GetParagraphSpacingAfter() == attr2.GetParagraphSpacingAfter() &&
4416 attr1.GetParagraphSpacingBefore() == attr2.GetParagraphSpacingBefore() &&
4417 attr1.GetBulletStyle() == attr2.GetBulletStyle() &&
4418 attr1.GetBulletNumber() == attr2.GetBulletNumber() &&
4419 attr1.GetBulletSymbol() == attr2.GetBulletSymbol() &&
4420 attr1.GetCharacterStyleName() == attr2.GetCharacterStyleName() &&
4421 attr1.GetParagraphStyleName() == attr2.GetParagraphStyleName());
4422}
4423
4424bool wxTextAttrEq(const wxTextAttrEx& attr1, const wxRichTextAttr& attr2)
4425{
4426 return (
4427 attr1.GetTextColour() == attr2.GetTextColour() &&
4428 attr1.GetBackgroundColour() == attr2.GetBackgroundColour() &&
4429 attr1.GetFont().GetPointSize() == attr2.GetFontSize() &&
4430 attr1.GetFont().GetStyle() == attr2.GetFontStyle() &&
4431 attr1.GetFont().GetWeight() == attr2.GetFontWeight() &&
4432 attr1.GetFont().GetFaceName() == attr2.GetFontFaceName() &&
4433 attr1.GetFont().GetUnderlined() == attr2.GetFontUnderlined() &&
4434 attr1.GetAlignment() == attr2.GetAlignment() &&
4435 attr1.GetLeftIndent() == attr2.GetLeftIndent() &&
4436 attr1.GetRightIndent() == attr2.GetRightIndent() &&
4437 attr1.GetLeftSubIndent() == attr2.GetLeftSubIndent() &&
4438 attr1.GetTabs().GetCount() == attr2.GetTabs().GetCount() && // heuristic
4439 attr1.GetLineSpacing() == attr2.GetLineSpacing() &&
4440 attr1.GetParagraphSpacingAfter() == attr2.GetParagraphSpacingAfter() &&
4441 attr1.GetParagraphSpacingBefore() == attr2.GetParagraphSpacingBefore() &&
4442 attr1.GetBulletStyle() == attr2.GetBulletStyle() &&
4443 attr1.GetBulletNumber() == attr2.GetBulletNumber() &&
4444 attr1.GetBulletSymbol() == attr2.GetBulletSymbol() &&
4445 attr1.GetCharacterStyleName() == attr2.GetCharacterStyleName() &&
4446 attr1.GetParagraphStyleName() == attr2.GetParagraphStyleName());
4447}
4448
4449/// Compare two attribute objects, but take into account the flags
4450/// specifying attributes of interest.
4451bool wxTextAttrEqPartial(const wxTextAttrEx& attr1, const wxTextAttrEx& attr2, int flags)
4452{
4453 if ((flags & wxTEXT_ATTR_TEXT_COLOUR) && attr1.GetTextColour() != attr2.GetTextColour())
4454 return false;
4455
4456 if ((flags & wxTEXT_ATTR_BACKGROUND_COLOUR) && attr1.GetBackgroundColour() != attr2.GetBackgroundColour())
4457 return false;
4458
4459 if ((flags & wxTEXT_ATTR_FONT_FACE) && attr1.GetFont().Ok() && attr2.GetFont().Ok() &&
4460 attr1.GetFont().GetFaceName() != attr2.GetFont().GetFaceName())
4461 return false;
4462
4463 if ((flags & wxTEXT_ATTR_FONT_SIZE) && attr1.GetFont().Ok() && attr2.GetFont().Ok() &&
4464 attr1.GetFont().GetPointSize() != attr2.GetFont().GetPointSize())
4465 return false;
4466
4467 if ((flags & wxTEXT_ATTR_FONT_WEIGHT) && attr1.GetFont().Ok() && attr2.GetFont().Ok() &&
4468 attr1.GetFont().GetWeight() != attr2.GetFont().GetWeight())
4469 return false;
4470
4471 if ((flags & wxTEXT_ATTR_FONT_ITALIC) && attr1.GetFont().Ok() && attr2.GetFont().Ok() &&
4472 attr1.GetFont().GetStyle() != attr2.GetFont().GetStyle())
4473 return false;
4474
4475 if ((flags & wxTEXT_ATTR_FONT_UNDERLINE) && attr1.GetFont().Ok() && attr2.GetFont().Ok() &&
4476 attr1.GetFont().GetUnderlined() != attr2.GetFont().GetUnderlined())
4477 return false;
4478
4479 if ((flags & wxTEXT_ATTR_ALIGNMENT) && attr1.GetAlignment() != attr2.GetAlignment())
4480 return false;
4481
4482 if ((flags & wxTEXT_ATTR_LEFT_INDENT) &&
4483 ((attr1.GetLeftIndent() != attr2.GetLeftIndent()) || (attr1.GetLeftSubIndent() != attr2.GetLeftSubIndent())))
4484 return false;
4485
4486 if ((flags & wxTEXT_ATTR_RIGHT_INDENT) &&
4487 (attr1.GetRightIndent() != attr2.GetRightIndent()))
4488 return false;
4489
4d551ad5 4490 if ((flags & wxTEXT_ATTR_PARA_SPACING_AFTER) &&
5d7836c4
JS
4491 (attr1.GetParagraphSpacingAfter() != attr2.GetParagraphSpacingAfter()))
4492 return false;
4493
4d551ad5 4494 if ((flags & wxTEXT_ATTR_PARA_SPACING_BEFORE) &&
5d7836c4
JS
4495 (attr1.GetParagraphSpacingBefore() != attr2.GetParagraphSpacingBefore()))
4496 return false;
4497
4d551ad5 4498 if ((flags & wxTEXT_ATTR_LINE_SPACING) &&
5d7836c4
JS
4499 (attr1.GetLineSpacing() != attr2.GetLineSpacing()))
4500 return false;
4501
4d551ad5 4502 if ((flags & wxTEXT_ATTR_CHARACTER_STYLE_NAME) &&
5d7836c4
JS
4503 (attr1.GetCharacterStyleName() != attr2.GetCharacterStyleName()))
4504 return false;
4505
4d551ad5 4506 if ((flags & wxTEXT_ATTR_PARAGRAPH_STYLE_NAME) &&
5d7836c4
JS
4507 (attr1.GetParagraphStyleName() != attr2.GetParagraphStyleName()))
4508 return false;
4509
4510 if ((flags & wxTEXT_ATTR_BULLET_STYLE) &&
4511 (attr1.GetBulletStyle() != attr2.GetBulletStyle()))
4512 return false;
4513
4514 if ((flags & wxTEXT_ATTR_BULLET_NUMBER) &&
4515 (attr1.GetBulletNumber() != attr2.GetBulletNumber()))
4516 return false;
4517
4518 if ((flags & wxTEXT_ATTR_BULLET_SYMBOL) &&
4519 (attr1.GetBulletSymbol() != attr2.GetBulletSymbol()))
4520 return false;
4521
4522/* TODO
4523 if ((flags & wxTEXT_ATTR_TABS) &&
4524 return false;
4525*/
4526
4527 return true;
4528}
4529
4530bool wxTextAttrEqPartial(const wxTextAttrEx& attr1, const wxRichTextAttr& attr2, int flags)
4531{
4532 if ((flags & wxTEXT_ATTR_TEXT_COLOUR) && attr1.GetTextColour() != attr2.GetTextColour())
4533 return false;
4534
4535 if ((flags & wxTEXT_ATTR_BACKGROUND_COLOUR) && attr1.GetBackgroundColour() != attr2.GetBackgroundColour())
4536 return false;
4537
4538 if ((flags & (wxTEXT_ATTR_FONT)) && !attr1.GetFont().Ok())
4539 return false;
4540
4541 if ((flags & wxTEXT_ATTR_FONT_FACE) && attr1.GetFont().Ok() &&
4542 attr1.GetFont().GetFaceName() != attr2.GetFontFaceName())
4543 return false;
4544
4545 if ((flags & wxTEXT_ATTR_FONT_SIZE) && attr1.GetFont().Ok() &&
4546 attr1.GetFont().GetPointSize() != attr2.GetFontSize())
4547 return false;
4548
4549 if ((flags & wxTEXT_ATTR_FONT_WEIGHT) && attr1.GetFont().Ok() &&
4550 attr1.GetFont().GetWeight() != attr2.GetFontWeight())
4551 return false;
4552
4553 if ((flags & wxTEXT_ATTR_FONT_ITALIC) && attr1.GetFont().Ok() &&
4554 attr1.GetFont().GetStyle() != attr2.GetFontStyle())
4555 return false;
4556
4557 if ((flags & wxTEXT_ATTR_FONT_UNDERLINE) && attr1.GetFont().Ok() &&
4558 attr1.GetFont().GetUnderlined() != attr2.GetFontUnderlined())
4559 return false;
4560
4561 if ((flags & wxTEXT_ATTR_ALIGNMENT) && attr1.GetAlignment() != attr2.GetAlignment())
4562 return false;
4563
4564 if ((flags & wxTEXT_ATTR_LEFT_INDENT) &&
4565 ((attr1.GetLeftIndent() != attr2.GetLeftIndent()) || (attr1.GetLeftSubIndent() != attr2.GetLeftSubIndent())))
4566 return false;
4567
4568 if ((flags & wxTEXT_ATTR_RIGHT_INDENT) &&
4569 (attr1.GetRightIndent() != attr2.GetRightIndent()))
4570 return false;
4571
4d551ad5 4572 if ((flags & wxTEXT_ATTR_PARA_SPACING_AFTER) &&
5d7836c4
JS
4573 (attr1.GetParagraphSpacingAfter() != attr2.GetParagraphSpacingAfter()))
4574 return false;
4575
4d551ad5 4576 if ((flags & wxTEXT_ATTR_PARA_SPACING_BEFORE) &&
5d7836c4
JS
4577 (attr1.GetParagraphSpacingBefore() != attr2.GetParagraphSpacingBefore()))
4578 return false;
4579
4d551ad5 4580 if ((flags & wxTEXT_ATTR_LINE_SPACING) &&
5d7836c4
JS
4581 (attr1.GetLineSpacing() != attr2.GetLineSpacing()))
4582 return false;
4583
4d551ad5 4584 if ((flags & wxTEXT_ATTR_CHARACTER_STYLE_NAME) &&
5d7836c4
JS
4585 (attr1.GetCharacterStyleName() != attr2.GetCharacterStyleName()))
4586 return false;
4587
4d551ad5 4588 if ((flags & wxTEXT_ATTR_PARAGRAPH_STYLE_NAME) &&
5d7836c4
JS
4589 (attr1.GetParagraphStyleName() != attr2.GetParagraphStyleName()))
4590 return false;
4591
4592 if ((flags & wxTEXT_ATTR_BULLET_STYLE) &&
4593 (attr1.GetBulletStyle() != attr2.GetBulletStyle()))
4594 return false;
4595
4596 if ((flags & wxTEXT_ATTR_BULLET_NUMBER) &&
4597 (attr1.GetBulletNumber() != attr2.GetBulletNumber()))
4598 return false;
4599
4600 if ((flags & wxTEXT_ATTR_BULLET_SYMBOL) &&
4601 (attr1.GetBulletSymbol() != attr2.GetBulletSymbol()))
4602 return false;
4603
4604/* TODO
4605 if ((flags & wxTEXT_ATTR_TABS) &&
4606 return false;
4607*/
4608
4609 return true;
4610}
4611
4612
4613/// Apply one style to another
4614bool wxRichTextApplyStyle(wxTextAttrEx& destStyle, const wxTextAttrEx& style)
4615{
4616 // Whole font
4617 if (style.GetFont().Ok() && ((style.GetFlags() & (wxTEXT_ATTR_FONT)) == (wxTEXT_ATTR_FONT)))
4618 destStyle.SetFont(style.GetFont());
4619 else if (style.GetFont().Ok())
4620 {
4621 wxFont font = destStyle.GetFont();
4622
4623 if (style.GetFlags() & wxTEXT_ATTR_FONT_FACE)
4624 font.SetFaceName(style.GetFont().GetFaceName());
4625
4626 if (style.GetFlags() & wxTEXT_ATTR_FONT_SIZE)
4627 font.SetPointSize(style.GetFont().GetPointSize());
4628
4629 if (style.GetFlags() & wxTEXT_ATTR_FONT_ITALIC)
4630 font.SetStyle(style.GetFont().GetStyle());
4631
4632 if (style.GetFlags() & wxTEXT_ATTR_FONT_WEIGHT)
4633 font.SetWeight(style.GetFont().GetWeight());
4634
4635 if (style.GetFlags() & wxTEXT_ATTR_FONT_UNDERLINE)
4636 font.SetUnderlined(style.GetFont().GetUnderlined());
4637
4638 if (font != destStyle.GetFont())
4639 destStyle.SetFont(font);
4640 }
4641
4642 if ( style.GetTextColour().Ok() && style.HasTextColour())
4643 destStyle.SetTextColour(style.GetTextColour());
4644
4645 if ( style.GetBackgroundColour().Ok() && style.HasBackgroundColour())
4646 destStyle.SetBackgroundColour(style.GetBackgroundColour());
4647
4648 if (style.HasAlignment())
4649 destStyle.SetAlignment(style.GetAlignment());
4650
4651 if (style.HasTabs())
4652 destStyle.SetTabs(style.GetTabs());
4653
4654 if (style.HasLeftIndent())
4655 destStyle.SetLeftIndent(style.GetLeftIndent(), style.GetLeftSubIndent());
4656
4657 if (style.HasRightIndent())
4658 destStyle.SetRightIndent(style.GetRightIndent());
4659
4660 if (style.HasParagraphSpacingAfter())
4661 destStyle.SetParagraphSpacingAfter(style.GetParagraphSpacingAfter());
4662
4663 if (style.HasParagraphSpacingBefore())
4664 destStyle.SetParagraphSpacingBefore(style.GetParagraphSpacingBefore());
4665
4666 if (style.HasLineSpacing())
4667 destStyle.SetLineSpacing(style.GetLineSpacing());
4668
4669 if (style.HasCharacterStyleName())
4670 destStyle.SetCharacterStyleName(style.GetCharacterStyleName());
4671
4672 if (style.HasParagraphStyleName())
4673 destStyle.SetParagraphStyleName(style.GetParagraphStyleName());
4674
4675 if (style.HasBulletStyle())
4676 {
4677 destStyle.SetBulletStyle(style.GetBulletStyle());
4678 destStyle.SetBulletSymbol(style.GetBulletSymbol());
4679 }
4680
4681 if (style.HasBulletNumber())
4682 destStyle.SetBulletNumber(style.GetBulletNumber());
4683
4684 return true;
4685}
4686
4687bool wxRichTextApplyStyle(wxRichTextAttr& destStyle, const wxTextAttrEx& style)
4688{
4689 wxTextAttrEx destStyle2;
4690 destStyle.CopyTo(destStyle2);
4691 wxRichTextApplyStyle(destStyle2, style);
4692 destStyle = destStyle2;
4693 return true;
4694}
4695
4696bool wxRichTextApplyStyle(wxTextAttrEx& destStyle, const wxRichTextAttr& style)
4697{
7fe8059f 4698
5d7836c4
JS
4699 // Whole font. Avoiding setting individual attributes if possible, since
4700 // it recreates the font each time.
4701 if ((style.GetFlags() & (wxTEXT_ATTR_FONT)) == (wxTEXT_ATTR_FONT))
4702 {
4703 destStyle.SetFont(wxFont(style.GetFontSize(), destStyle.GetFont().Ok() ? destStyle.GetFont().GetFamily() : wxDEFAULT,
4704 style.GetFontStyle(), style.GetFontWeight(), style.GetFontUnderlined(), style.GetFontFaceName()));
4705 }
4706 else if (style.GetFlags() & (wxTEXT_ATTR_FONT))
4707 {
4708 wxFont font = destStyle.GetFont();
4709
4710 if (style.GetFlags() & wxTEXT_ATTR_FONT_FACE)
4711 font.SetFaceName(style.GetFontFaceName());
7fe8059f 4712
5d7836c4
JS
4713 if (style.GetFlags() & wxTEXT_ATTR_FONT_SIZE)
4714 font.SetPointSize(style.GetFontSize());
7fe8059f 4715
5d7836c4
JS
4716 if (style.GetFlags() & wxTEXT_ATTR_FONT_ITALIC)
4717 font.SetStyle(style.GetFontStyle());
7fe8059f 4718
5d7836c4
JS
4719 if (style.GetFlags() & wxTEXT_ATTR_FONT_WEIGHT)
4720 font.SetWeight(style.GetFontWeight());
7fe8059f 4721
5d7836c4
JS
4722 if (style.GetFlags() & wxTEXT_ATTR_FONT_UNDERLINE)
4723 font.SetUnderlined(style.GetFontUnderlined());
4724
4725 if (font != destStyle.GetFont())
4726 destStyle.SetFont(font);
4727 }
4728
4729 if ( style.GetTextColour().Ok() && style.HasTextColour())
4730 destStyle.SetTextColour(style.GetTextColour());
4731
4732 if ( style.GetBackgroundColour().Ok() && style.HasBackgroundColour())
4733 destStyle.SetBackgroundColour(style.GetBackgroundColour());
4734
4735 if (style.HasAlignment())
4736 destStyle.SetAlignment(style.GetAlignment());
4737
4738 if (style.HasTabs())
4739 destStyle.SetTabs(style.GetTabs());
4740
4741 if (style.HasLeftIndent())
4742 destStyle.SetLeftIndent(style.GetLeftIndent(), style.GetLeftSubIndent());
4743
4744 if (style.HasRightIndent())
4745 destStyle.SetRightIndent(style.GetRightIndent());
4746
4747 if (style.HasParagraphSpacingAfter())
4748 destStyle.SetParagraphSpacingAfter(style.GetParagraphSpacingAfter());
4749
4750 if (style.HasParagraphSpacingBefore())
4751 destStyle.SetParagraphSpacingBefore(style.GetParagraphSpacingBefore());
4752
4753 if (style.HasLineSpacing())
4754 destStyle.SetLineSpacing(style.GetLineSpacing());
4755
4756 if (style.HasCharacterStyleName())
4757 destStyle.SetCharacterStyleName(style.GetCharacterStyleName());
4758
4759 if (style.HasParagraphStyleName())
4760 destStyle.SetParagraphStyleName(style.GetParagraphStyleName());
4761
4762 if (style.HasBulletStyle())
4763 {
4764 destStyle.SetBulletStyle(style.GetBulletStyle());
4765 destStyle.SetBulletSymbol(style.GetBulletSymbol());
4766 }
4767
4768 if (style.HasBulletNumber())
4769 destStyle.SetBulletNumber(style.GetBulletNumber());
4770
4771 return true;
4772}
4773
4774
4775/*!
4776 * wxRichTextAttr stores attributes without a wxFont object, so is a much more
4777 * efficient way to query styles.
4778 */
4779
4780// ctors
4781wxRichTextAttr::wxRichTextAttr(const wxColour& colText,
4782 const wxColour& colBack,
4783 wxTextAttrAlignment alignment): m_textAlignment(alignment), m_colText(colText), m_colBack(colBack)
4784{
4785 Init();
4786
4787 if (m_colText.Ok()) m_flags |= wxTEXT_ATTR_TEXT_COLOUR;
4788 if (m_colBack.Ok()) m_flags |= wxTEXT_ATTR_BACKGROUND_COLOUR;
4789 if (alignment != wxTEXT_ALIGNMENT_DEFAULT)
4790 m_flags |= wxTEXT_ATTR_ALIGNMENT;
4791}
4792
4793wxRichTextAttr::wxRichTextAttr(const wxTextAttrEx& attr)
4794{
4795 Init();
4796
4797 (*this) = attr;
4798}
4799
4800// operations
4801void wxRichTextAttr::Init()
4802{
4803 m_textAlignment = wxTEXT_ALIGNMENT_DEFAULT;
4804 m_flags = 0;
4805 m_leftIndent = 0;
4806 m_leftSubIndent = 0;
4807 m_rightIndent = 0;
4808
4809 m_fontSize = 12;
4810 m_fontStyle = wxNORMAL;
4811 m_fontWeight = wxNORMAL;
4812 m_fontUnderlined = false;
4813
4814 m_paragraphSpacingAfter = 0;
4815 m_paragraphSpacingBefore = 0;
4816 m_lineSpacing = 0;
4817 m_bulletStyle = wxTEXT_ATTR_BULLET_STYLE_NONE;
4818 m_bulletNumber = 0;
4819 m_bulletSymbol = wxT('*');
4820}
4821
4822// operators
4823void wxRichTextAttr::operator= (const wxRichTextAttr& attr)
4824{
4825 m_colText = attr.m_colText;
4826 m_colBack = attr.m_colBack;
4827 m_textAlignment = attr.m_textAlignment;
4828 m_leftIndent = attr.m_leftIndent;
4829 m_leftSubIndent = attr.m_leftSubIndent;
4830 m_rightIndent = attr.m_rightIndent;
4831 m_tabs = attr.m_tabs;
4832 m_flags = attr.m_flags;
4833
4834 m_fontSize = attr.m_fontSize;
4835 m_fontStyle = attr.m_fontStyle;
4836 m_fontWeight = attr.m_fontWeight;
4837 m_fontUnderlined = attr.m_fontUnderlined;
4838 m_fontFaceName = attr.m_fontFaceName;
4839
4840 m_paragraphSpacingAfter = attr.m_paragraphSpacingAfter;
4841 m_paragraphSpacingBefore = attr.m_paragraphSpacingBefore;
4842 m_lineSpacing = attr.m_lineSpacing;
4843 m_characterStyleName = attr.m_characterStyleName;
4844 m_paragraphStyleName = attr.m_paragraphStyleName;
4845 m_bulletStyle = attr.m_bulletStyle;
4846 m_bulletNumber = attr.m_bulletNumber;
4847 m_bulletSymbol = attr.m_bulletSymbol;
4848}
4849
4850// operators
4851void wxRichTextAttr::operator= (const wxTextAttrEx& attr)
4852{
4853 m_colText = attr.GetTextColour();
4854 m_colBack = attr.GetBackgroundColour();
4855 m_textAlignment = attr.GetAlignment();
4856 m_leftIndent = attr.GetLeftIndent();
4857 m_leftSubIndent = attr.GetLeftSubIndent();
4858 m_rightIndent = attr.GetRightIndent();
4859 m_tabs = attr.GetTabs();
4860 m_flags = attr.GetFlags();
4861
4862 m_paragraphSpacingAfter = attr.GetParagraphSpacingAfter();
4863 m_paragraphSpacingBefore = attr.GetParagraphSpacingBefore();
4864 m_lineSpacing = attr.GetLineSpacing();
4865 m_characterStyleName = attr.GetCharacterStyleName();
4866 m_paragraphStyleName = attr.GetParagraphStyleName();
4867
4868 if (attr.GetFont().Ok())
4869 GetFontAttributes(attr.GetFont());
4870}
4871
4872// Making a wxTextAttrEx object.
4873wxRichTextAttr::operator wxTextAttrEx () const
4874{
4875 wxTextAttrEx attr;
4876 CopyTo(attr);
4877 return attr;
4878}
4879
4880// Copy to a wxTextAttr
4881void wxRichTextAttr::CopyTo(wxTextAttrEx& attr) const
4882{
4883 attr.SetTextColour(GetTextColour());
4884 attr.SetBackgroundColour(GetBackgroundColour());
4885 attr.SetAlignment(GetAlignment());
4886 attr.SetTabs(GetTabs());
4887 attr.SetLeftIndent(GetLeftIndent(), GetLeftSubIndent());
4888 attr.SetRightIndent(GetRightIndent());
4889 attr.SetFont(CreateFont());
4890 attr.SetFlags(GetFlags()); // Important: set after SetFont, since SetFont sets flags
4891
4892 attr.SetParagraphSpacingAfter(m_paragraphSpacingAfter);
4893 attr.SetParagraphSpacingBefore(m_paragraphSpacingBefore);
4894 attr.SetLineSpacing(m_lineSpacing);
4895 attr.SetBulletStyle(m_bulletStyle);
4896 attr.SetBulletNumber(m_bulletNumber);
4897 attr.SetBulletSymbol(m_bulletSymbol);
4898 attr.SetCharacterStyleName(m_characterStyleName);
4899 attr.SetParagraphStyleName(m_paragraphStyleName);
4900
4901}
4902
4903// Create font from font attributes.
4904wxFont wxRichTextAttr::CreateFont() const
4905{
4906 wxFont font(m_fontSize, wxDEFAULT, m_fontStyle, m_fontWeight, m_fontUnderlined, m_fontFaceName);
ff2baa25
JS
4907#ifdef __WXMAC__
4908 font.SetNoAntiAliasing(true);
4909#endif
5d7836c4
JS
4910 return font;
4911}
4912
4913// Get attributes from font.
4914bool wxRichTextAttr::GetFontAttributes(const wxFont& font)
4915{
4916 if (!font.Ok())
4917 return false;
4918
4919 m_fontSize = font.GetPointSize();
4920 m_fontStyle = font.GetStyle();
4921 m_fontWeight = font.GetWeight();
4922 m_fontUnderlined = font.GetUnderlined();
4923 m_fontFaceName = font.GetFaceName();
4924
4925 return true;
4926}
4927
05b4fddf
JS
4928wxRichTextAttr wxRichTextAttr::Combine(const wxRichTextAttr& attr,
4929 const wxRichTextAttr& attrDef,
4930 const wxTextCtrlBase *text)
61399247 4931{
05b4fddf
JS
4932 wxColour colFg = attr.GetTextColour();
4933 if ( !colFg.Ok() )
4934 {
4935 colFg = attrDef.GetTextColour();
4936
4937 if ( text && !colFg.Ok() )
4938 colFg = text->GetForegroundColour();
4939 }
4940
4941 wxColour colBg = attr.GetBackgroundColour();
4942 if ( !colBg.Ok() )
4943 {
4944 colBg = attrDef.GetBackgroundColour();
4945
4946 if ( text && !colBg.Ok() )
4947 colBg = text->GetBackgroundColour();
4948 }
4949
4950 wxRichTextAttr newAttr(colFg, colBg);
4951
4952 if (attr.HasWeight())
4953 newAttr.SetFontWeight(attr.GetFontWeight());
61399247 4954
05b4fddf
JS
4955 if (attr.HasSize())
4956 newAttr.SetFontSize(attr.GetFontSize());
61399247 4957
05b4fddf
JS
4958 if (attr.HasItalic())
4959 newAttr.SetFontStyle(attr.GetFontStyle());
61399247 4960
05b4fddf
JS
4961 if (attr.HasUnderlined())
4962 newAttr.SetFontUnderlined(attr.GetFontUnderlined());
61399247 4963
05b4fddf
JS
4964 if (attr.HasFaceName())
4965 newAttr.SetFontFaceName(attr.GetFontFaceName());
61399247 4966
05b4fddf
JS
4967 if (attr.HasAlignment())
4968 newAttr.SetAlignment(attr.GetAlignment());
4969 else if (attrDef.HasAlignment())
4970 newAttr.SetAlignment(attrDef.GetAlignment());
4971
4972 if (attr.HasTabs())
4973 newAttr.SetTabs(attr.GetTabs());
4974 else if (attrDef.HasTabs())
4975 newAttr.SetTabs(attrDef.GetTabs());
4976
4977 if (attr.HasLeftIndent())
4978 newAttr.SetLeftIndent(attr.GetLeftIndent(), attr.GetLeftSubIndent());
4979 else if (attrDef.HasLeftIndent())
4980 newAttr.SetLeftIndent(attrDef.GetLeftIndent(), attr.GetLeftSubIndent());
4981
4982 if (attr.HasRightIndent())
4983 newAttr.SetRightIndent(attr.GetRightIndent());
4984 else if (attrDef.HasRightIndent())
4985 newAttr.SetRightIndent(attrDef.GetRightIndent());
61399247 4986
05b4fddf 4987 // NEW ATTRIBUTES
61399247 4988
05b4fddf
JS
4989 if (attr.HasParagraphSpacingAfter())
4990 newAttr.SetParagraphSpacingAfter(attr.GetParagraphSpacingAfter());
61399247 4991
05b4fddf
JS
4992 if (attr.HasParagraphSpacingBefore())
4993 newAttr.SetParagraphSpacingBefore(attr.GetParagraphSpacingBefore());
4994
4995 if (attr.HasLineSpacing())
4996 newAttr.SetLineSpacing(attr.GetLineSpacing());
61399247 4997
05b4fddf
JS
4998 if (attr.HasCharacterStyleName())
4999 newAttr.SetCharacterStyleName(attr.GetCharacterStyleName());
61399247 5000
05b4fddf
JS
5001 if (attr.HasParagraphStyleName())
5002 newAttr.SetParagraphStyleName(attr.GetParagraphStyleName());
61399247 5003
05b4fddf
JS
5004 if (attr.HasBulletStyle())
5005 newAttr.SetBulletStyle(attr.GetBulletStyle());
5006
5007 if (attr.HasBulletNumber())
5008 newAttr.SetBulletNumber(attr.GetBulletNumber());
61399247 5009
05b4fddf
JS
5010 if (attr.HasBulletSymbol())
5011 newAttr.SetBulletSymbol(attr.GetBulletSymbol());
61399247 5012
05b4fddf
JS
5013 return newAttr;
5014}
5015
5d7836c4
JS
5016/*!
5017 * wxTextAttrEx is an extended version of wxTextAttr with more paragraph attributes.
5018 */
5019
5020wxTextAttrEx::wxTextAttrEx(const wxTextAttrEx& attr): wxTextAttr(attr)
5021{
5022 m_paragraphSpacingAfter = attr.m_paragraphSpacingAfter;
5023 m_paragraphSpacingBefore = attr.m_paragraphSpacingBefore;
5024 m_lineSpacing = attr.m_lineSpacing;
5025 m_paragraphStyleName = attr.m_paragraphStyleName;
5026 m_characterStyleName = attr.m_characterStyleName;
5027 m_bulletStyle = attr.m_bulletStyle;
5028 m_bulletNumber = attr.m_bulletNumber;
5029 m_bulletSymbol = attr.m_bulletSymbol;
5030}
5031
5032// Initialise this object.
5033void wxTextAttrEx::Init()
5034{
5035 m_paragraphSpacingAfter = 0;
5036 m_paragraphSpacingBefore = 0;
5037 m_lineSpacing = 0;
5038 m_bulletStyle = wxTEXT_ATTR_BULLET_STYLE_NONE;
5039 m_bulletNumber = 0;
5040 m_bulletSymbol = 0;
5041 m_bulletSymbol = wxT('*');
5042}
5043
5044// Assignment from a wxTextAttrEx object
5045void wxTextAttrEx::operator= (const wxTextAttrEx& attr)
5046{
5047 wxTextAttr::operator= (attr);
5048
5049 m_paragraphSpacingAfter = attr.m_paragraphSpacingAfter;
5050 m_paragraphSpacingBefore = attr.m_paragraphSpacingBefore;
5051 m_lineSpacing = attr.m_lineSpacing;
5052 m_characterStyleName = attr.m_characterStyleName;
5053 m_paragraphStyleName = attr.m_paragraphStyleName;
5054 m_bulletStyle = attr.m_bulletStyle;
5055 m_bulletNumber = attr.m_bulletNumber;
5056 m_bulletSymbol = attr.m_bulletSymbol;
5057}
5058
5059// Assignment from a wxTextAttr object.
5060void wxTextAttrEx::operator= (const wxTextAttr& attr)
5061{
5062 wxTextAttr::operator= (attr);
5063}
5064
05b4fddf
JS
5065wxTextAttrEx wxTextAttrEx::CombineEx(const wxTextAttrEx& attr,
5066 const wxTextAttrEx& attrDef,
5067 const wxTextCtrlBase *text)
61399247 5068{
05b4fddf
JS
5069 wxTextAttrEx newAttr;
5070
5071 // If attr specifies the complete font, just use that font, overriding all
5072 // default font attributes.
5073 if ((attr.GetFlags() & wxTEXT_ATTR_FONT) == wxTEXT_ATTR_FONT)
5074 newAttr.SetFont(attr.GetFont());
5075 else
5076 {
5077 // First find the basic, default font
5078 long flags = 0;
5079
5080 wxFont font;
5081 if (attrDef.HasFont())
5082 {
5083 flags = (attrDef.GetFlags() & wxTEXT_ATTR_FONT);
5084 font = attrDef.GetFont();
5085 }
5086 else
5087 {
5088 if (text)
5089 font = text->GetFont();
5090
5091 // We leave flags at 0 because no font attributes have been specified yet
5092 }
5093 if (!font.Ok())
5094 font = *wxNORMAL_FONT;
5095
5096 // Otherwise, if there are font attributes in attr, apply them
5097 if (attr.HasFont())
5098 {
5099 if (attr.HasSize())
5100 {
5101 flags |= wxTEXT_ATTR_FONT_SIZE;
5102 font.SetPointSize(attr.GetFont().GetPointSize());
5103 }
5104 if (attr.HasItalic())
5105 {
5106 flags |= wxTEXT_ATTR_FONT_ITALIC;;
5107 font.SetStyle(attr.GetFont().GetStyle());
5108 }
5109 if (attr.HasWeight())
5110 {
5111 flags |= wxTEXT_ATTR_FONT_WEIGHT;
5112 font.SetWeight(attr.GetFont().GetWeight());
5113 }
5114 if (attr.HasFaceName())
5115 {
5116 flags |= wxTEXT_ATTR_FONT_FACE;
5117 font.SetFaceName(attr.GetFont().GetFaceName());
5118 }
5119 if (attr.HasUnderlined())
5120 {
5121 flags |= wxTEXT_ATTR_FONT_UNDERLINE;
5122 font.SetUnderlined(attr.GetFont().GetUnderlined());
5123 }
5124 newAttr.SetFont(font);
5125 newAttr.SetFlags(newAttr.GetFlags()|flags);
5126 }
5127 }
5128
5129 // TODO: should really check we are specifying these in the flags,
5130 // before setting them, as per above; or we will set them willy-nilly.
5131 // However, we should also check whether this is the intention
5132 // as per wxTextAttr::Combine, i.e. always to have valid colours
5133 // in the style.
5134 wxColour colFg = attr.GetTextColour();
5135 if ( !colFg.Ok() )
5136 {
5137 colFg = attrDef.GetTextColour();
5138
5139 if ( text && !colFg.Ok() )
5140 colFg = text->GetForegroundColour();
5141 }
5142
5143 wxColour colBg = attr.GetBackgroundColour();
5144 if ( !colBg.Ok() )
5145 {
5146 colBg = attrDef.GetBackgroundColour();
5147
5148 if ( text && !colBg.Ok() )
5149 colBg = text->GetBackgroundColour();
5150 }
5151
5152 newAttr.SetTextColour(colFg);
5153 newAttr.SetBackgroundColour(colBg);
5154
5155 if (attr.HasAlignment())
5156 newAttr.SetAlignment(attr.GetAlignment());
5157 else if (attrDef.HasAlignment())
5158 newAttr.SetAlignment(attrDef.GetAlignment());
5159
5160 if (attr.HasTabs())
5161 newAttr.SetTabs(attr.GetTabs());
5162 else if (attrDef.HasTabs())
5163 newAttr.SetTabs(attrDef.GetTabs());
5164
5165 if (attr.HasLeftIndent())
5166 newAttr.SetLeftIndent(attr.GetLeftIndent(), attr.GetLeftSubIndent());
5167 else if (attrDef.HasLeftIndent())
5168 newAttr.SetLeftIndent(attrDef.GetLeftIndent(), attr.GetLeftSubIndent());
5169
5170 if (attr.HasRightIndent())
5171 newAttr.SetRightIndent(attr.GetRightIndent());
5172 else if (attrDef.HasRightIndent())
5173 newAttr.SetRightIndent(attrDef.GetRightIndent());
61399247 5174
05b4fddf 5175 // NEW ATTRIBUTES
61399247 5176
05b4fddf
JS
5177 if (attr.HasParagraphSpacingAfter())
5178 newAttr.SetParagraphSpacingAfter(attr.GetParagraphSpacingAfter());
61399247 5179
05b4fddf
JS
5180 if (attr.HasParagraphSpacingBefore())
5181 newAttr.SetParagraphSpacingBefore(attr.GetParagraphSpacingBefore());
5182
5183 if (attr.HasLineSpacing())
5184 newAttr.SetLineSpacing(attr.GetLineSpacing());
61399247 5185
05b4fddf
JS
5186 if (attr.HasCharacterStyleName())
5187 newAttr.SetCharacterStyleName(attr.GetCharacterStyleName());
61399247 5188
05b4fddf
JS
5189 if (attr.HasParagraphStyleName())
5190 newAttr.SetParagraphStyleName(attr.GetParagraphStyleName());
61399247 5191
05b4fddf
JS
5192 if (attr.HasBulletStyle())
5193 newAttr.SetBulletStyle(attr.GetBulletStyle());
5194
5195 if (attr.HasBulletNumber())
5196 newAttr.SetBulletNumber(attr.GetBulletNumber());
61399247 5197
05b4fddf
JS
5198 if (attr.HasBulletSymbol())
5199 newAttr.SetBulletSymbol(attr.GetBulletSymbol());
61399247 5200
05b4fddf
JS
5201 return newAttr;
5202}
5203
5204
5d7836c4
JS
5205/*!
5206 * wxRichTextFileHandler
5207 * Base class for file handlers
5208 */
5209
5210IMPLEMENT_CLASS(wxRichTextFileHandler, wxObject)
5211
7fe8059f 5212#if wxUSE_STREAMS
5d7836c4
JS
5213bool wxRichTextFileHandler::LoadFile(wxRichTextBuffer *buffer, const wxString& filename)
5214{
5215 wxFFileInputStream stream(filename);
5216 if (stream.Ok())
5217 return LoadFile(buffer, stream);
caae22fb
WS
5218
5219 return false;
5d7836c4
JS
5220}
5221
5222bool wxRichTextFileHandler::SaveFile(wxRichTextBuffer *buffer, const wxString& filename)
5223{
5224 wxFFileOutputStream stream(filename);
5225 if (stream.Ok())
5226 return SaveFile(buffer, stream);
caae22fb
WS
5227
5228 return false;
5d7836c4 5229}
7fe8059f 5230#endif // wxUSE_STREAMS
5d7836c4
JS
5231
5232/// Can we handle this filename (if using files)? By default, checks the extension.
5233bool wxRichTextFileHandler::CanHandle(const wxString& filename) const
5234{
5235 wxString path, file, ext;
5236 wxSplitPath(filename, & path, & file, & ext);
5237
5238 return (ext.Lower() == GetExtension());
5239}
5240
5241/*!
5242 * wxRichTextTextHandler
5243 * Plain text handler
5244 */
5245
5246IMPLEMENT_CLASS(wxRichTextPlainTextHandler, wxRichTextFileHandler)
5247
5248#if wxUSE_STREAMS
7fe8059f 5249bool wxRichTextPlainTextHandler::DoLoadFile(wxRichTextBuffer *buffer, wxInputStream& stream)
5d7836c4
JS
5250{
5251 if (!stream.IsOk())
5252 return false;
5253
5254 wxString str;
1e967276 5255 int lastCh = 0;
5d7836c4
JS
5256
5257 while (!stream.Eof())
5258 {
e191ee87 5259 int ch = stream.GetC();
5d7836c4 5260
169adfa9
JS
5261 if (!stream.Eof())
5262 {
5263 if (ch == 10 && lastCh != 13)
5264 str += wxT('\n');
39a1c2f2 5265
169adfa9
JS
5266 if (ch > 0 && ch != 10)
5267 str += wxChar(ch);
39a1c2f2 5268
169adfa9
JS
5269 lastCh = ch;
5270 }
5d7836c4
JS
5271 }
5272
5273 buffer->Clear();
5274 buffer->AddParagraphs(str);
5275 buffer->UpdateRanges();
5276
5277 return true;
5278
5279}
5280
7fe8059f 5281bool wxRichTextPlainTextHandler::DoSaveFile(wxRichTextBuffer *buffer, wxOutputStream& stream)
5d7836c4
JS
5282{
5283 if (!stream.IsOk())
5284 return false;
5285
5286 wxString text = buffer->GetText();
5287 wxCharBuffer buf = text.ToAscii();
5288
28f92d74 5289 stream.Write((const char*) buf, text.length());
5d7836c4
JS
5290 return true;
5291}
7fe8059f 5292#endif // wxUSE_STREAMS
5d7836c4
JS
5293
5294/*
5295 * Stores information about an image, in binary in-memory form
5296 */
5297
5298wxRichTextImageBlock::wxRichTextImageBlock()
5299{
5300 Init();
5301}
5302
5303wxRichTextImageBlock::wxRichTextImageBlock(const wxRichTextImageBlock& block):wxObject()
5304{
5305 Init();
5306 Copy(block);
5307}
5308
5309wxRichTextImageBlock::~wxRichTextImageBlock()
5310{
5311 if (m_data)
5312 {
5313 delete[] m_data;
5314 m_data = NULL;
5315 }
5316}
5317
5318void wxRichTextImageBlock::Init()
5319{
5320 m_data = NULL;
5321 m_dataSize = 0;
5322 m_imageType = -1;
5323}
5324
5325void wxRichTextImageBlock::Clear()
5326{
b01ca8b6 5327 delete[] m_data;
5d7836c4
JS
5328 m_data = NULL;
5329 m_dataSize = 0;
5330 m_imageType = -1;
5331}
5332
5333
5334// Load the original image into a memory block.
5335// If the image is not a JPEG, we must convert it into a JPEG
5336// to conserve space.
5337// If it's not a JPEG we can make use of 'image', already scaled, so we don't have to
5338// load the image a 2nd time.
5339
5340bool wxRichTextImageBlock::MakeImageBlock(const wxString& filename, int imageType, wxImage& image, bool convertToJPEG)
5341{
5342 m_imageType = imageType;
5343
5344 wxString filenameToRead(filename);
7fe8059f 5345 bool removeFile = false;
5d7836c4
JS
5346
5347 if (imageType == -1)
7fe8059f 5348 return false; // Could not determine image type
5d7836c4
JS
5349
5350 if ((imageType != wxBITMAP_TYPE_JPEG) && convertToJPEG)
5351 {
5352 wxString tempFile;
5353 bool success = wxGetTempFileName(_("image"), tempFile) ;
5354
5355 wxASSERT(success);
5356
5357 wxUnusedVar(success);
5358
5359 image.SaveFile(tempFile, wxBITMAP_TYPE_JPEG);
5360 filenameToRead = tempFile;
7fe8059f 5361 removeFile = true;
5d7836c4
JS
5362
5363 m_imageType = wxBITMAP_TYPE_JPEG;
5364 }
5365 wxFile file;
5366 if (!file.Open(filenameToRead))
7fe8059f 5367 return false;
5d7836c4
JS
5368
5369 m_dataSize = (size_t) file.Length();
5370 file.Close();
5371
5372 if (m_data)
5373 delete[] m_data;
5374 m_data = ReadBlock(filenameToRead, m_dataSize);
5375
5376 if (removeFile)
5377 wxRemoveFile(filenameToRead);
5378
5379 return (m_data != NULL);
5380}
5381
5382// Make an image block from the wxImage in the given
5383// format.
5384bool wxRichTextImageBlock::MakeImageBlock(wxImage& image, int imageType, int quality)
5385{
5386 m_imageType = imageType;
5387 image.SetOption(wxT("quality"), quality);
5388
5389 if (imageType == -1)
7fe8059f 5390 return false; // Could not determine image type
5d7836c4
JS
5391
5392 wxString tempFile;
5393 bool success = wxGetTempFileName(_("image"), tempFile) ;
7fe8059f 5394
5d7836c4
JS
5395 wxASSERT(success);
5396 wxUnusedVar(success);
7fe8059f 5397
5d7836c4
JS
5398 if (!image.SaveFile(tempFile, m_imageType))
5399 {
5400 if (wxFileExists(tempFile))
5401 wxRemoveFile(tempFile);
7fe8059f 5402 return false;
5d7836c4
JS
5403 }
5404
5405 wxFile file;
5406 if (!file.Open(tempFile))
7fe8059f 5407 return false;
5d7836c4
JS
5408
5409 m_dataSize = (size_t) file.Length();
5410 file.Close();
5411
5412 if (m_data)
5413 delete[] m_data;
5414 m_data = ReadBlock(tempFile, m_dataSize);
5415
5416 wxRemoveFile(tempFile);
5417
5418 return (m_data != NULL);
5419}
5420
5421
5422// Write to a file
5423bool wxRichTextImageBlock::Write(const wxString& filename)
5424{
5425 return WriteBlock(filename, m_data, m_dataSize);
5426}
5427
5428void wxRichTextImageBlock::Copy(const wxRichTextImageBlock& block)
5429{
5430 m_imageType = block.m_imageType;
5431 if (m_data)
5432 {
5433 delete[] m_data;
5434 m_data = NULL;
5435 }
5436 m_dataSize = block.m_dataSize;
5437 if (m_dataSize == 0)
5438 return;
5439
5440 m_data = new unsigned char[m_dataSize];
5441 unsigned int i;
5442 for (i = 0; i < m_dataSize; i++)
5443 m_data[i] = block.m_data[i];
5444}
5445
5446//// Operators
5447void wxRichTextImageBlock::operator=(const wxRichTextImageBlock& block)
5448{
5449 Copy(block);
5450}
5451
5452// Load a wxImage from the block
5453bool wxRichTextImageBlock::Load(wxImage& image)
5454{
5455 if (!m_data)
7fe8059f 5456 return false;
5d7836c4
JS
5457
5458 // Read in the image.
5459#if 1
5460 wxMemoryInputStream mstream(m_data, m_dataSize);
5461 bool success = image.LoadFile(mstream, GetImageType());
5462#else
5463 wxString tempFile;
5464 bool success = wxGetTempFileName(_("image"), tempFile) ;
5465 wxASSERT(success);
5466
5467 if (!WriteBlock(tempFile, m_data, m_dataSize))
5468 {
7fe8059f 5469 return false;
5d7836c4
JS
5470 }
5471 success = image.LoadFile(tempFile, GetImageType());
5472 wxRemoveFile(tempFile);
5473#endif
5474
5475 return success;
5476}
5477
5478// Write data in hex to a stream
5479bool wxRichTextImageBlock::WriteHex(wxOutputStream& stream)
5480{
5481 wxString hex;
5482 int i;
5483 for (i = 0; i < (int) m_dataSize; i++)
5484 {
5485 hex = wxDecToHex(m_data[i]);
5486 wxCharBuffer buf = hex.ToAscii();
7fe8059f 5487
28f92d74 5488 stream.Write((const char*) buf, hex.length());
5d7836c4
JS
5489 }
5490
5491 return true;
5492}
5493
5494// Read data in hex from a stream
5495bool wxRichTextImageBlock::ReadHex(wxInputStream& stream, int length, int imageType)
5496{
5497 int dataSize = length/2;
5498
5499 if (m_data)
5500 delete[] m_data;
5501
5502 wxString str(wxT(" "));
5503 m_data = new unsigned char[dataSize];
5504 int i;
5505 for (i = 0; i < dataSize; i ++)
5506 {
5507 str[0] = stream.GetC();
5508 str[1] = stream.GetC();
5509
7fe8059f 5510 m_data[i] = (unsigned char)wxHexToDec(str);
5d7836c4
JS
5511 }
5512
5513 m_dataSize = dataSize;
5514 m_imageType = imageType;
5515
5516 return true;
5517}
5518
5519
5520// Allocate and read from stream as a block of memory
5521unsigned char* wxRichTextImageBlock::ReadBlock(wxInputStream& stream, size_t size)
5522{
5523 unsigned char* block = new unsigned char[size];
5524 if (!block)
5525 return NULL;
5526
5527 stream.Read(block, size);
5528
5529 return block;
5530}
5531
5532unsigned char* wxRichTextImageBlock::ReadBlock(const wxString& filename, size_t size)
5533{
5534 wxFileInputStream stream(filename);
5535 if (!stream.Ok())
5536 return NULL;
5537
5538 return ReadBlock(stream, size);
5539}
5540
5541// Write memory block to stream
5542bool wxRichTextImageBlock::WriteBlock(wxOutputStream& stream, unsigned char* block, size_t size)
5543{
5544 stream.Write((void*) block, size);
5545 return stream.IsOk();
5546
5547}
5548
5549// Write memory block to file
5550bool wxRichTextImageBlock::WriteBlock(const wxString& filename, unsigned char* block, size_t size)
5551{
5552 wxFileOutputStream outStream(filename);
5553 if (!outStream.Ok())
7fe8059f 5554 return false;
5d7836c4
JS
5555
5556 return WriteBlock(outStream, block, size);
5557}
5558
5559#endif
5560 // wxUSE_RICHTEXT