]> git.saurik.com Git - wxWidgets.git/blob - src/richtext/richtextbuffer.cpp
be0a26852c268fd5ca150609eeb94d3853a58472
[wxWidgets.git] / src / richtext / richtextbuffer.cpp
1 /////////////////////////////////////////////////////////////////////////////
2 // Name: src/richtext/richtextbuffer.cpp
3 // Purpose: Buffer for wxRichTextCtrl
4 // Author: Julian Smart
5 // Modified by:
6 // Created: 2005-09-30
7 // RCS-ID: $Id$
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__
16 #pragma hdrstop
17 #endif
18
19 #if wxUSE_RICHTEXT
20
21 #include "wx/richtext/richtextbuffer.h"
22
23 #ifndef WX_PRECOMP
24 #include "wx/dc.h"
25 #include "wx/intl.h"
26 #include "wx/log.h"
27 #include "wx/dataobj.h"
28 #include "wx/module.h"
29 #endif
30
31 #include "wx/settings.h"
32 #include "wx/filename.h"
33 #include "wx/clipbrd.h"
34 #include "wx/wfstream.h"
35 #include "wx/mstream.h"
36 #include "wx/sstream.h"
37 #include "wx/textfile.h"
38 #include "wx/hashmap.h"
39
40 #include "wx/richtext/richtextctrl.h"
41 #include "wx/richtext/richtextstyles.h"
42
43 #include "wx/listimpl.cpp"
44
45 WX_DEFINE_LIST(wxRichTextObjectList)
46 WX_DEFINE_LIST(wxRichTextLineList)
47
48 // Switch off if the platform doesn't like it for some reason
49 #define wxRICHTEXT_USE_OPTIMIZED_DRAWING 1
50
51 // Use GetPartialTextExtents for platforms that support it natively
52 #define wxRICHTEXT_USE_PARTIAL_TEXT_EXTENTS 1
53
54 const wxChar wxRichTextLineBreakChar = (wxChar) 29;
55
56 // Helpers for efficiency
57
58 inline void wxCheckSetFont(wxDC& dc, const wxFont& font)
59 {
60 const wxFont& font1 = dc.GetFont();
61 if (font1.IsOk() && font.IsOk())
62 {
63 if (font1.GetPointSize() == font.GetPointSize() &&
64 font1.GetFamily() == font.GetFamily() &&
65 font1.GetStyle() == font.GetStyle() &&
66 font1.GetWeight() == font.GetWeight() &&
67 font1.GetUnderlined() == font.GetUnderlined() &&
68 font1.GetFaceName() == font.GetFaceName())
69 return;
70 }
71 dc.SetFont(font);
72 }
73
74 inline void wxCheckSetPen(wxDC& dc, const wxPen& pen)
75 {
76 const wxPen& pen1 = dc.GetPen();
77 if (pen1.IsOk() && pen.IsOk())
78 {
79 if (pen1.GetWidth() == pen.GetWidth() &&
80 pen1.GetStyle() == pen.GetStyle() &&
81 pen1.GetColour() == pen.GetColour())
82 return;
83 }
84 dc.SetPen(pen);
85 }
86
87 inline void wxCheckSetBrush(wxDC& dc, const wxBrush& brush)
88 {
89 const wxBrush& brush1 = dc.GetBrush();
90 if (brush1.IsOk() && brush.IsOk())
91 {
92 if (brush1.GetStyle() == brush.GetStyle() &&
93 brush1.GetColour() == brush.GetColour())
94 return;
95 }
96 dc.SetBrush(brush);
97 }
98
99 /*!
100 * wxRichTextObject
101 * This is the base for drawable objects.
102 */
103
104 IMPLEMENT_CLASS(wxRichTextObject, wxObject)
105
106 wxRichTextObject::wxRichTextObject(wxRichTextObject* parent)
107 {
108 m_dirty = false;
109 m_refCount = 1;
110 m_parent = parent;
111 m_leftMargin = 0;
112 m_rightMargin = 0;
113 m_topMargin = 0;
114 m_bottomMargin = 0;
115 m_descent = 0;
116 }
117
118 wxRichTextObject::~wxRichTextObject()
119 {
120 }
121
122 void wxRichTextObject::Dereference()
123 {
124 m_refCount --;
125 if (m_refCount <= 0)
126 delete this;
127 }
128
129 /// Copy
130 void wxRichTextObject::Copy(const wxRichTextObject& obj)
131 {
132 m_size = obj.m_size;
133 m_pos = obj.m_pos;
134 m_dirty = obj.m_dirty;
135 m_range = obj.m_range;
136 m_attributes = obj.m_attributes;
137 m_descent = obj.m_descent;
138 }
139
140 void wxRichTextObject::SetMargins(int margin)
141 {
142 m_leftMargin = m_rightMargin = m_topMargin = m_bottomMargin = margin;
143 }
144
145 void wxRichTextObject::SetMargins(int leftMargin, int rightMargin, int topMargin, int bottomMargin)
146 {
147 m_leftMargin = leftMargin;
148 m_rightMargin = rightMargin;
149 m_topMargin = topMargin;
150 m_bottomMargin = bottomMargin;
151 }
152
153 // Convert units in tenths of a millimetre to device units
154 int wxRichTextObject::ConvertTenthsMMToPixels(wxDC& dc, int units)
155 {
156 int p = ConvertTenthsMMToPixels(dc.GetPPI().x, units);
157
158 // Unscale
159 wxRichTextBuffer* buffer = GetBuffer();
160 if (buffer)
161 p = (int) ((double)p / buffer->GetScale());
162 return p;
163 }
164
165 // Convert units in tenths of a millimetre to device units
166 int wxRichTextObject::ConvertTenthsMMToPixels(int ppi, int units)
167 {
168 // There are ppi pixels in 254.1 "1/10 mm"
169
170 double pixels = ((double) units * (double)ppi) / 254.1;
171
172 return (int) pixels;
173 }
174
175 /// Dump to output stream for debugging
176 void wxRichTextObject::Dump(wxTextOutputStream& stream)
177 {
178 stream << GetClassInfo()->GetClassName() << wxT("\n");
179 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");
180 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");
181 }
182
183 /// Gets the containing buffer
184 wxRichTextBuffer* wxRichTextObject::GetBuffer() const
185 {
186 const wxRichTextObject* obj = this;
187 while (obj && !obj->IsKindOf(CLASSINFO(wxRichTextBuffer)))
188 obj = obj->GetParent();
189 return wxDynamicCast(obj, wxRichTextBuffer);
190 }
191
192 /*!
193 * wxRichTextCompositeObject
194 * This is the base for drawable objects.
195 */
196
197 IMPLEMENT_CLASS(wxRichTextCompositeObject, wxRichTextObject)
198
199 wxRichTextCompositeObject::wxRichTextCompositeObject(wxRichTextObject* parent):
200 wxRichTextObject(parent)
201 {
202 }
203
204 wxRichTextCompositeObject::~wxRichTextCompositeObject()
205 {
206 DeleteChildren();
207 }
208
209 /// Get the nth child
210 wxRichTextObject* wxRichTextCompositeObject::GetChild(size_t n) const
211 {
212 wxASSERT ( n < m_children.GetCount() );
213
214 return m_children.Item(n)->GetData();
215 }
216
217 /// Append a child, returning the position
218 size_t wxRichTextCompositeObject::AppendChild(wxRichTextObject* child)
219 {
220 m_children.Append(child);
221 child->SetParent(this);
222 return m_children.GetCount() - 1;
223 }
224
225 /// Insert the child in front of the given object, or at the beginning
226 bool wxRichTextCompositeObject::InsertChild(wxRichTextObject* child, wxRichTextObject* inFrontOf)
227 {
228 if (inFrontOf)
229 {
230 wxRichTextObjectList::compatibility_iterator node = m_children.Find(inFrontOf);
231 m_children.Insert(node, child);
232 }
233 else
234 m_children.Insert(child);
235 child->SetParent(this);
236
237 return true;
238 }
239
240 /// Delete the child
241 bool wxRichTextCompositeObject::RemoveChild(wxRichTextObject* child, bool deleteChild)
242 {
243 wxRichTextObjectList::compatibility_iterator node = m_children.Find(child);
244 if (node)
245 {
246 wxRichTextObject* obj = node->GetData();
247 m_children.Erase(node);
248 if (deleteChild)
249 delete obj;
250
251 return true;
252 }
253 return false;
254 }
255
256 /// Delete all children
257 bool wxRichTextCompositeObject::DeleteChildren()
258 {
259 wxRichTextObjectList::compatibility_iterator node = m_children.GetFirst();
260 while (node)
261 {
262 wxRichTextObjectList::compatibility_iterator oldNode = node;
263
264 wxRichTextObject* child = node->GetData();
265 child->Dereference(); // Only delete if reference count is zero
266
267 node = node->GetNext();
268 m_children.Erase(oldNode);
269 }
270
271 return true;
272 }
273
274 /// Get the child count
275 size_t wxRichTextCompositeObject::GetChildCount() const
276 {
277 return m_children.GetCount();
278 }
279
280 /// Copy
281 void wxRichTextCompositeObject::Copy(const wxRichTextCompositeObject& obj)
282 {
283 wxRichTextObject::Copy(obj);
284
285 DeleteChildren();
286
287 wxRichTextObjectList::compatibility_iterator node = obj.m_children.GetFirst();
288 while (node)
289 {
290 wxRichTextObject* child = node->GetData();
291 wxRichTextObject* newChild = child->Clone();
292 newChild->SetParent(this);
293 m_children.Append(newChild);
294
295 node = node->GetNext();
296 }
297 }
298
299 /// Hit-testing: returns a flag indicating hit test details, plus
300 /// information about position
301 int wxRichTextCompositeObject::HitTest(wxDC& dc, const wxPoint& pt, long& textPosition)
302 {
303 wxRichTextObjectList::compatibility_iterator node = m_children.GetFirst();
304 while (node)
305 {
306 wxRichTextObject* child = node->GetData();
307
308 int ret = child->HitTest(dc, pt, textPosition);
309 if (ret != wxRICHTEXT_HITTEST_NONE)
310 return ret;
311
312 node = node->GetNext();
313 }
314
315 textPosition = GetRange().GetEnd()-1;
316 return wxRICHTEXT_HITTEST_AFTER|wxRICHTEXT_HITTEST_OUTSIDE;
317 }
318
319 /// Finds the absolute position and row height for the given character position
320 bool wxRichTextCompositeObject::FindPosition(wxDC& dc, long index, wxPoint& pt, int* height, bool forceLineStart)
321 {
322 wxRichTextObjectList::compatibility_iterator node = m_children.GetFirst();
323 while (node)
324 {
325 wxRichTextObject* child = node->GetData();
326
327 if (child->FindPosition(dc, index, pt, height, forceLineStart))
328 return true;
329
330 node = node->GetNext();
331 }
332
333 return false;
334 }
335
336 /// Calculate range
337 void wxRichTextCompositeObject::CalculateRange(long start, long& end)
338 {
339 long current = start;
340 long lastEnd = current;
341
342 wxRichTextObjectList::compatibility_iterator node = m_children.GetFirst();
343 while (node)
344 {
345 wxRichTextObject* child = node->GetData();
346 long childEnd = 0;
347
348 child->CalculateRange(current, childEnd);
349 lastEnd = childEnd;
350
351 current = childEnd + 1;
352
353 node = node->GetNext();
354 }
355
356 end = lastEnd;
357
358 // An object with no children has zero length
359 if (m_children.GetCount() == 0)
360 end --;
361
362 m_range.SetRange(start, end);
363 }
364
365 /// Delete range from layout.
366 bool wxRichTextCompositeObject::DeleteRange(const wxRichTextRange& range)
367 {
368 wxRichTextObjectList::compatibility_iterator node = m_children.GetFirst();
369
370 while (node)
371 {
372 wxRichTextObject* obj = (wxRichTextObject*) node->GetData();
373 wxRichTextObjectList::compatibility_iterator next = node->GetNext();
374
375 // Delete the range in each paragraph
376
377 // When a chunk has been deleted, internally the content does not
378 // now match the ranges.
379 // However, so long as deletion is not done on the same object twice this is OK.
380 // If you may delete content from the same object twice, recalculate
381 // the ranges inbetween DeleteRange calls by calling CalculateRanges, and
382 // adjust the range you're deleting accordingly.
383
384 if (!obj->GetRange().IsOutside(range))
385 {
386 obj->DeleteRange(range);
387
388 // Delete an empty object, or paragraph within this range.
389 if (obj->IsEmpty() ||
390 (range.GetStart() <= obj->GetRange().GetStart() && range.GetEnd() >= obj->GetRange().GetEnd()))
391 {
392 // An empty paragraph has length 1, so won't be deleted unless the
393 // whole range is deleted.
394 RemoveChild(obj, true);
395 }
396 }
397
398 node = next;
399 }
400
401 return true;
402 }
403
404 /// Get any text in this object for the given range
405 wxString wxRichTextCompositeObject::GetTextForRange(const wxRichTextRange& range) const
406 {
407 wxString text;
408 wxRichTextObjectList::compatibility_iterator node = m_children.GetFirst();
409 while (node)
410 {
411 wxRichTextObject* child = node->GetData();
412 wxRichTextRange childRange = range;
413 if (!child->GetRange().IsOutside(range))
414 {
415 childRange.LimitTo(child->GetRange());
416
417 wxString childText = child->GetTextForRange(childRange);
418
419 text += childText;
420 }
421 node = node->GetNext();
422 }
423
424 return text;
425 }
426
427 /// Recursively merge all pieces that can be merged.
428 bool wxRichTextCompositeObject::Defragment()
429 {
430 wxRichTextObjectList::compatibility_iterator node = m_children.GetFirst();
431 while (node)
432 {
433 wxRichTextObject* child = node->GetData();
434 wxRichTextCompositeObject* composite = wxDynamicCast(child, wxRichTextCompositeObject);
435 if (composite)
436 composite->Defragment();
437
438 if (node->GetNext())
439 {
440 wxRichTextObject* nextChild = node->GetNext()->GetData();
441 if (child->CanMerge(nextChild) && child->Merge(nextChild))
442 {
443 nextChild->Dereference();
444 m_children.Erase(node->GetNext());
445
446 // Don't set node -- we'll see if we can merge again with the next
447 // child.
448 }
449 else
450 node = node->GetNext();
451 }
452 else
453 node = node->GetNext();
454 }
455
456 return true;
457 }
458
459 /// Dump to output stream for debugging
460 void wxRichTextCompositeObject::Dump(wxTextOutputStream& stream)
461 {
462 wxRichTextObjectList::compatibility_iterator node = m_children.GetFirst();
463 while (node)
464 {
465 wxRichTextObject* child = node->GetData();
466 child->Dump(stream);
467 node = node->GetNext();
468 }
469 }
470
471
472 /*!
473 * wxRichTextBox
474 * This defines a 2D space to lay out objects
475 */
476
477 IMPLEMENT_DYNAMIC_CLASS(wxRichTextBox, wxRichTextCompositeObject)
478
479 wxRichTextBox::wxRichTextBox(wxRichTextObject* parent):
480 wxRichTextCompositeObject(parent)
481 {
482 }
483
484 /// Draw the item
485 bool wxRichTextBox::Draw(wxDC& dc, const wxRichTextRange& range, const wxRichTextRange& selectionRange, const wxRect& WXUNUSED(rect), int descent, int style)
486 {
487 wxRichTextObjectList::compatibility_iterator node = m_children.GetFirst();
488 while (node)
489 {
490 wxRichTextObject* child = node->GetData();
491
492 wxRect childRect = wxRect(child->GetPosition(), child->GetCachedSize());
493 child->Draw(dc, range, selectionRange, childRect, descent, style);
494
495 node = node->GetNext();
496 }
497 return true;
498 }
499
500 /// Lay the item out
501 bool wxRichTextBox::Layout(wxDC& dc, const wxRect& rect, int style)
502 {
503 wxRichTextObjectList::compatibility_iterator node = m_children.GetFirst();
504 while (node)
505 {
506 wxRichTextObject* child = node->GetData();
507 child->Layout(dc, rect, style);
508
509 node = node->GetNext();
510 }
511 m_dirty = false;
512 return true;
513 }
514
515 /// Get/set the size for the given range. Assume only has one child.
516 bool wxRichTextBox::GetRangeSize(const wxRichTextRange& range, wxSize& size, int& descent, wxDC& dc, int flags, wxPoint position, wxArrayInt* partialExtents) const
517 {
518 wxRichTextObjectList::compatibility_iterator node = m_children.GetFirst();
519 if (node)
520 {
521 wxRichTextObject* child = node->GetData();
522 return child->GetRangeSize(range, size, descent, dc, flags, position, partialExtents);
523 }
524 else
525 return false;
526 }
527
528 /// Copy
529 void wxRichTextBox::Copy(const wxRichTextBox& obj)
530 {
531 wxRichTextCompositeObject::Copy(obj);
532 }
533
534
535 /*!
536 * wxRichTextParagraphLayoutBox
537 * This box knows how to lay out paragraphs.
538 */
539
540 IMPLEMENT_DYNAMIC_CLASS(wxRichTextParagraphLayoutBox, wxRichTextBox)
541
542 wxRichTextParagraphLayoutBox::wxRichTextParagraphLayoutBox(wxRichTextObject* parent):
543 wxRichTextBox(parent)
544 {
545 Init();
546 }
547
548 /// Initialize the object.
549 void wxRichTextParagraphLayoutBox::Init()
550 {
551 m_ctrl = NULL;
552
553 // For now, assume is the only box and has no initial size.
554 m_range = wxRichTextRange(0, -1);
555
556 m_invalidRange.SetRange(-1, -1);
557 m_leftMargin = 4;
558 m_rightMargin = 4;
559 m_topMargin = 4;
560 m_bottomMargin = 4;
561 m_partialParagraph = false;
562 }
563
564 /// Draw the item
565 bool wxRichTextParagraphLayoutBox::Draw(wxDC& dc, const wxRichTextRange& range, const wxRichTextRange& selectionRange, const wxRect& rect, int descent, int style)
566 {
567 wxRichTextObjectList::compatibility_iterator node = m_children.GetFirst();
568 while (node)
569 {
570 wxRichTextParagraph* child = wxDynamicCast(node->GetData(), wxRichTextParagraph);
571 wxASSERT (child != NULL);
572
573 if (child && !child->GetRange().IsOutside(range))
574 {
575 wxRect childRect(child->GetPosition(), child->GetCachedSize());
576
577 if (((style & wxRICHTEXT_DRAW_IGNORE_CACHE) == 0) && childRect.GetTop() > rect.GetBottom())
578 {
579 // Stop drawing
580 break;
581 }
582 else if (((style & wxRICHTEXT_DRAW_IGNORE_CACHE) == 0) && childRect.GetBottom() < rect.GetTop())
583 {
584 // Skip
585 }
586 else
587 child->Draw(dc, range, selectionRange, childRect, descent, style);
588 }
589
590 node = node->GetNext();
591 }
592 return true;
593 }
594
595 /// Lay the item out
596 bool wxRichTextParagraphLayoutBox::Layout(wxDC& dc, const wxRect& rect, int style)
597 {
598 wxRect availableSpace;
599 bool formatRect = (style & wxRICHTEXT_LAYOUT_SPECIFIED_RECT) == wxRICHTEXT_LAYOUT_SPECIFIED_RECT;
600
601 // If only laying out a specific area, the passed rect has a different meaning:
602 // the visible part of the buffer. This is used in wxRichTextCtrl::OnSize,
603 // so that during a size, only the visible part will be relaid out, or
604 // it would take too long causing flicker. As an approximation, we assume that
605 // everything up to the start of the visible area is laid out correctly.
606 if (formatRect)
607 {
608 availableSpace = wxRect(0 + m_leftMargin,
609 0 + m_topMargin,
610 rect.width - m_leftMargin - m_rightMargin,
611 rect.height);
612
613 // Invalidate the part of the buffer from the first visible line
614 // to the end. If other parts of the buffer are currently invalid,
615 // then they too will be taken into account if they are above
616 // the visible point.
617 long startPos = 0;
618 wxRichTextLine* line = GetLineAtYPosition(rect.y);
619 if (line)
620 startPos = line->GetAbsoluteRange().GetStart();
621
622 Invalidate(wxRichTextRange(startPos, GetRange().GetEnd()));
623 }
624 else
625 availableSpace = wxRect(rect.x + m_leftMargin,
626 rect.y + m_topMargin,
627 rect.width - m_leftMargin - m_rightMargin,
628 rect.height - m_topMargin - m_bottomMargin);
629
630 int maxWidth = 0;
631
632 wxRichTextObjectList::compatibility_iterator node = m_children.GetFirst();
633
634 bool layoutAll = true;
635
636 // Get invalid range, rounding to paragraph start/end.
637 wxRichTextRange invalidRange = GetInvalidRange(true);
638
639 if (invalidRange == wxRICHTEXT_NONE && !formatRect)
640 return true;
641
642 if (invalidRange == wxRICHTEXT_ALL)
643 layoutAll = true;
644 else // If we know what range is affected, start laying out from that point on.
645 if (invalidRange.GetStart() >= GetRange().GetStart())
646 {
647 wxRichTextParagraph* firstParagraph = GetParagraphAtPosition(invalidRange.GetStart());
648 if (firstParagraph)
649 {
650 wxRichTextObjectList::compatibility_iterator firstNode = m_children.Find(firstParagraph);
651 wxRichTextObjectList::compatibility_iterator previousNode;
652 if ( firstNode )
653 previousNode = firstNode->GetPrevious();
654 if (firstNode)
655 {
656 if (previousNode)
657 {
658 wxRichTextParagraph* previousParagraph = wxDynamicCast(previousNode->GetData(), wxRichTextParagraph);
659 availableSpace.y = previousParagraph->GetPosition().y + previousParagraph->GetCachedSize().y;
660 }
661
662 // Now we're going to start iterating from the first affected paragraph.
663 node = firstNode;
664
665 layoutAll = false;
666 }
667 }
668 }
669
670 // A way to force speedy rest-of-buffer layout (the 'else' below)
671 bool forceQuickLayout = false;
672
673 while (node)
674 {
675 // Assume this box only contains paragraphs
676
677 wxRichTextParagraph* child = wxDynamicCast(node->GetData(), wxRichTextParagraph);
678 wxCHECK_MSG( child, false, _T("Unknown object in layout") );
679
680 // TODO: what if the child hasn't been laid out (e.g. involved in Undo) but still has 'old' lines
681 if ( !forceQuickLayout &&
682 (layoutAll ||
683 child->GetLines().IsEmpty() ||
684 !child->GetRange().IsOutside(invalidRange)) )
685 {
686 child->Layout(dc, availableSpace, style);
687
688 // Layout must set the cached size
689 availableSpace.y += child->GetCachedSize().y;
690 maxWidth = wxMax(maxWidth, child->GetCachedSize().x);
691
692 // If we're just formatting the visible part of the buffer,
693 // and we're now past the bottom of the window, start quick
694 // layout.
695 if (formatRect && child->GetPosition().y > rect.GetBottom())
696 forceQuickLayout = true;
697 }
698 else
699 {
700 // We're outside the immediately affected range, so now let's just
701 // move everything up or down. This assumes that all the children have previously
702 // been laid out and have wrapped line lists associated with them.
703 // TODO: check all paragraphs before the affected range.
704
705 int inc = availableSpace.y - child->GetPosition().y;
706
707 while (node)
708 {
709 wxRichTextParagraph* child = wxDynamicCast(node->GetData(), wxRichTextParagraph);
710 if (child)
711 {
712 if (child->GetLines().GetCount() == 0)
713 child->Layout(dc, availableSpace, style);
714 else
715 child->SetPosition(wxPoint(child->GetPosition().x, child->GetPosition().y + inc));
716
717 availableSpace.y += child->GetCachedSize().y;
718 maxWidth = wxMax(maxWidth, child->GetCachedSize().x);
719 }
720
721 node = node->GetNext();
722 }
723 break;
724 }
725
726 node = node->GetNext();
727 }
728
729 SetCachedSize(wxSize(maxWidth, availableSpace.y));
730
731 m_dirty = false;
732 m_invalidRange = wxRICHTEXT_NONE;
733
734 return true;
735 }
736
737 /// Copy
738 void wxRichTextParagraphLayoutBox::Copy(const wxRichTextParagraphLayoutBox& obj)
739 {
740 wxRichTextBox::Copy(obj);
741
742 m_partialParagraph = obj.m_partialParagraph;
743 m_defaultAttributes = obj.m_defaultAttributes;
744 }
745
746 /// Get/set the size for the given range.
747 bool wxRichTextParagraphLayoutBox::GetRangeSize(const wxRichTextRange& range, wxSize& size, int& descent, wxDC& dc, int flags, wxPoint position, wxArrayInt* WXUNUSED(partialExtents)) const
748 {
749 wxSize sz;
750
751 wxRichTextObjectList::compatibility_iterator startPara = wxRichTextObjectList::compatibility_iterator();
752 wxRichTextObjectList::compatibility_iterator endPara = wxRichTextObjectList::compatibility_iterator();
753
754 // First find the first paragraph whose starting position is within the range.
755 wxRichTextObjectList::compatibility_iterator node = m_children.GetFirst();
756 while (node)
757 {
758 // child is a paragraph
759 wxRichTextObject* child = node->GetData();
760 const wxRichTextRange& r = child->GetRange();
761
762 if (r.GetStart() <= range.GetStart() && r.GetEnd() >= range.GetStart())
763 {
764 startPara = node;
765 break;
766 }
767
768 node = node->GetNext();
769 }
770
771 // Next find the last paragraph containing part of the range
772 node = m_children.GetFirst();
773 while (node)
774 {
775 // child is a paragraph
776 wxRichTextObject* child = node->GetData();
777 const wxRichTextRange& r = child->GetRange();
778
779 if (r.GetStart() <= range.GetEnd() && r.GetEnd() >= range.GetEnd())
780 {
781 endPara = node;
782 break;
783 }
784
785 node = node->GetNext();
786 }
787
788 if (!startPara || !endPara)
789 return false;
790
791 // Now we can add up the sizes
792 for (node = startPara; node ; node = node->GetNext())
793 {
794 // child is a paragraph
795 wxRichTextObject* child = node->GetData();
796 const wxRichTextRange& childRange = child->GetRange();
797 wxRichTextRange rangeToFind = range;
798 rangeToFind.LimitTo(childRange);
799
800 wxSize childSize;
801
802 int childDescent = 0;
803 child->GetRangeSize(rangeToFind, childSize, childDescent, dc, flags, position);
804
805 descent = wxMax(childDescent, descent);
806
807 sz.x = wxMax(sz.x, childSize.x);
808 sz.y += childSize.y;
809
810 if (node == endPara)
811 break;
812 }
813
814 size = sz;
815
816 return true;
817 }
818
819 /// Get the paragraph at the given position
820 wxRichTextParagraph* wxRichTextParagraphLayoutBox::GetParagraphAtPosition(long pos, bool caretPosition) const
821 {
822 if (caretPosition)
823 pos ++;
824
825 // First find the first paragraph whose starting position is within the range.
826 wxRichTextObjectList::compatibility_iterator node = m_children.GetFirst();
827 while (node)
828 {
829 // child is a paragraph
830 wxRichTextParagraph* child = wxDynamicCast(node->GetData(), wxRichTextParagraph);
831 wxASSERT (child != NULL);
832
833 // Return first child in buffer if position is -1
834 // if (pos == -1)
835 // return child;
836
837 if (child->GetRange().Contains(pos))
838 return child;
839
840 node = node->GetNext();
841 }
842 return NULL;
843 }
844
845 /// Get the line at the given position
846 wxRichTextLine* wxRichTextParagraphLayoutBox::GetLineAtPosition(long pos, bool caretPosition) const
847 {
848 if (caretPosition)
849 pos ++;
850
851 // First find the first paragraph whose starting position is within the range.
852 wxRichTextObjectList::compatibility_iterator node = m_children.GetFirst();
853 while (node)
854 {
855 // child is a paragraph
856 wxRichTextParagraph* child = wxDynamicCast(node->GetData(), wxRichTextParagraph);
857 wxASSERT (child != NULL);
858
859 wxRichTextLineList::compatibility_iterator node2 = child->GetLines().GetFirst();
860 while (node2)
861 {
862 wxRichTextLine* line = node2->GetData();
863
864 wxRichTextRange range = line->GetAbsoluteRange();
865
866 if (range.Contains(pos) ||
867
868 // If the position is end-of-paragraph, then return the last line of
869 // of the paragraph.
870 (range.GetEnd() == child->GetRange().GetEnd()-1) && (pos == child->GetRange().GetEnd()))
871 return line;
872
873 node2 = node2->GetNext();
874 }
875
876 node = node->GetNext();
877 }
878
879 int lineCount = GetLineCount();
880 if (lineCount > 0)
881 return GetLineForVisibleLineNumber(lineCount-1);
882 else
883 return NULL;
884 }
885
886 /// Get the line at the given y pixel position, or the last line.
887 wxRichTextLine* wxRichTextParagraphLayoutBox::GetLineAtYPosition(int y) const
888 {
889 wxRichTextObjectList::compatibility_iterator node = m_children.GetFirst();
890 while (node)
891 {
892 wxRichTextParagraph* child = wxDynamicCast(node->GetData(), wxRichTextParagraph);
893 wxASSERT (child != NULL);
894
895 wxRichTextLineList::compatibility_iterator node2 = child->GetLines().GetFirst();
896 while (node2)
897 {
898 wxRichTextLine* line = node2->GetData();
899
900 wxRect rect(line->GetRect());
901
902 if (y <= rect.GetBottom())
903 return line;
904
905 node2 = node2->GetNext();
906 }
907
908 node = node->GetNext();
909 }
910
911 // Return last line
912 int lineCount = GetLineCount();
913 if (lineCount > 0)
914 return GetLineForVisibleLineNumber(lineCount-1);
915 else
916 return NULL;
917 }
918
919 /// Get the number of visible lines
920 int wxRichTextParagraphLayoutBox::GetLineCount() const
921 {
922 int count = 0;
923
924 wxRichTextObjectList::compatibility_iterator node = m_children.GetFirst();
925 while (node)
926 {
927 wxRichTextParagraph* child = wxDynamicCast(node->GetData(), wxRichTextParagraph);
928 wxASSERT (child != NULL);
929
930 count += child->GetLines().GetCount();
931 node = node->GetNext();
932 }
933 return count;
934 }
935
936
937 /// Get the paragraph for a given line
938 wxRichTextParagraph* wxRichTextParagraphLayoutBox::GetParagraphForLine(wxRichTextLine* line) const
939 {
940 return GetParagraphAtPosition(line->GetAbsoluteRange().GetStart());
941 }
942
943 /// Get the line size at the given position
944 wxSize wxRichTextParagraphLayoutBox::GetLineSizeAtPosition(long pos, bool caretPosition) const
945 {
946 wxRichTextLine* line = GetLineAtPosition(pos, caretPosition);
947 if (line)
948 {
949 return line->GetSize();
950 }
951 else
952 return wxSize(0, 0);
953 }
954
955
956 /// Convenience function to add a paragraph of text
957 wxRichTextRange wxRichTextParagraphLayoutBox::AddParagraph(const wxString& text, wxTextAttr* paraStyle)
958 {
959 // Don't use the base style, just the default style, and the base style will
960 // be combined at display time.
961 // Divide into paragraph and character styles.
962
963 wxTextAttr defaultCharStyle;
964 wxTextAttr defaultParaStyle;
965
966 // If the default style is a named paragraph style, don't apply any character formatting
967 // to the initial text string.
968 if (GetDefaultStyle().HasParagraphStyleName() && GetStyleSheet())
969 {
970 wxRichTextParagraphStyleDefinition* def = GetStyleSheet()->FindParagraphStyle(GetDefaultStyle().GetParagraphStyleName());
971 if (def)
972 defaultParaStyle = def->GetStyleMergedWithBase(GetStyleSheet());
973 }
974 else
975 wxRichTextSplitParaCharStyles(GetDefaultStyle(), defaultParaStyle, defaultCharStyle);
976
977 wxTextAttr* pStyle = paraStyle ? paraStyle : (wxTextAttr*) & defaultParaStyle;
978 wxTextAttr* cStyle = & defaultCharStyle;
979
980 wxRichTextParagraph* para = new wxRichTextParagraph(text, this, pStyle, cStyle);
981
982 AppendChild(para);
983
984 UpdateRanges();
985 SetDirty(true);
986
987 return para->GetRange();
988 }
989
990 /// Adds multiple paragraphs, based on newlines.
991 wxRichTextRange wxRichTextParagraphLayoutBox::AddParagraphs(const wxString& text, wxTextAttr* paraStyle)
992 {
993 // Don't use the base style, just the default style, and the base style will
994 // be combined at display time.
995 // Divide into paragraph and character styles.
996
997 wxTextAttr defaultCharStyle;
998 wxTextAttr defaultParaStyle;
999
1000 // If the default style is a named paragraph style, don't apply any character formatting
1001 // to the initial text string.
1002 if (GetDefaultStyle().HasParagraphStyleName() && GetStyleSheet())
1003 {
1004 wxRichTextParagraphStyleDefinition* def = GetStyleSheet()->FindParagraphStyle(GetDefaultStyle().GetParagraphStyleName());
1005 if (def)
1006 defaultParaStyle = def->GetStyleMergedWithBase(GetStyleSheet());
1007 }
1008 else
1009 wxRichTextSplitParaCharStyles(GetDefaultStyle(), defaultParaStyle, defaultCharStyle);
1010
1011 wxTextAttr* pStyle = paraStyle ? paraStyle : (wxTextAttr*) & defaultParaStyle;
1012 wxTextAttr* cStyle = & defaultCharStyle;
1013
1014 wxRichTextParagraph* firstPara = NULL;
1015 wxRichTextParagraph* lastPara = NULL;
1016
1017 wxRichTextRange range(-1, -1);
1018
1019 size_t i = 0;
1020 size_t len = text.length();
1021 wxString line;
1022 wxRichTextParagraph* para = new wxRichTextParagraph(wxEmptyString, this, pStyle, cStyle);
1023
1024 AppendChild(para);
1025
1026 firstPara = para;
1027 lastPara = para;
1028
1029 while (i < len)
1030 {
1031 wxChar ch = text[i];
1032 if (ch == wxT('\n') || ch == wxT('\r'))
1033 {
1034 if (i != (len-1))
1035 {
1036 wxRichTextPlainText* plainText = (wxRichTextPlainText*) para->GetChildren().GetFirst()->GetData();
1037 plainText->SetText(line);
1038
1039 para = new wxRichTextParagraph(wxEmptyString, this, pStyle, cStyle);
1040
1041 AppendChild(para);
1042
1043 lastPara = para;
1044 line = wxEmptyString;
1045 }
1046 }
1047 else
1048 line += ch;
1049
1050 i ++;
1051 }
1052
1053 if (!line.empty())
1054 {
1055 wxRichTextPlainText* plainText = (wxRichTextPlainText*) para->GetChildren().GetFirst()->GetData();
1056 plainText->SetText(line);
1057 }
1058
1059 UpdateRanges();
1060
1061 SetDirty(false);
1062
1063 return wxRichTextRange(firstPara->GetRange().GetStart(), lastPara->GetRange().GetEnd());
1064 }
1065
1066 /// Convenience function to add an image
1067 wxRichTextRange wxRichTextParagraphLayoutBox::AddImage(const wxImage& image, wxTextAttr* paraStyle)
1068 {
1069 // Don't use the base style, just the default style, and the base style will
1070 // be combined at display time.
1071 // Divide into paragraph and character styles.
1072
1073 wxTextAttr defaultCharStyle;
1074 wxTextAttr defaultParaStyle;
1075
1076 // If the default style is a named paragraph style, don't apply any character formatting
1077 // to the initial text string.
1078 if (GetDefaultStyle().HasParagraphStyleName() && GetStyleSheet())
1079 {
1080 wxRichTextParagraphStyleDefinition* def = GetStyleSheet()->FindParagraphStyle(GetDefaultStyle().GetParagraphStyleName());
1081 if (def)
1082 defaultParaStyle = def->GetStyleMergedWithBase(GetStyleSheet());
1083 }
1084 else
1085 wxRichTextSplitParaCharStyles(GetDefaultStyle(), defaultParaStyle, defaultCharStyle);
1086
1087 wxTextAttr* pStyle = paraStyle ? paraStyle : (wxTextAttr*) & defaultParaStyle;
1088 wxTextAttr* cStyle = & defaultCharStyle;
1089
1090 wxRichTextParagraph* para = new wxRichTextParagraph(this, pStyle);
1091 AppendChild(para);
1092 para->AppendChild(new wxRichTextImage(image, this, cStyle));
1093
1094 UpdateRanges();
1095 SetDirty(true);
1096
1097 return para->GetRange();
1098 }
1099
1100
1101 /// Insert fragment into this box at the given position. If partialParagraph is true,
1102 /// it is assumed that the last (or only) paragraph is just a piece of data with no paragraph
1103 /// marker.
1104
1105 bool wxRichTextParagraphLayoutBox::InsertFragment(long position, wxRichTextParagraphLayoutBox& fragment)
1106 {
1107 SetDirty(true);
1108
1109 // First, find the first paragraph whose starting position is within the range.
1110 wxRichTextParagraph* para = GetParagraphAtPosition(position);
1111 if (para)
1112 {
1113 wxTextAttrEx originalAttr = para->GetAttributes();
1114
1115 wxRichTextObjectList::compatibility_iterator node = m_children.Find(para);
1116
1117 // Now split at this position, returning the object to insert the new
1118 // ones in front of.
1119 wxRichTextObject* nextObject = para->SplitAt(position);
1120
1121 // Special case: partial paragraph, just one paragraph. Might be a small amount of
1122 // text, for example, so let's optimize.
1123
1124 if (fragment.GetPartialParagraph() && fragment.GetChildren().GetCount() == 1)
1125 {
1126 // Add the first para to this para...
1127 wxRichTextObjectList::compatibility_iterator firstParaNode = fragment.GetChildren().GetFirst();
1128 if (!firstParaNode)
1129 return false;
1130
1131 // Iterate through the fragment paragraph inserting the content into this paragraph.
1132 wxRichTextParagraph* firstPara = wxDynamicCast(firstParaNode->GetData(), wxRichTextParagraph);
1133 wxASSERT (firstPara != NULL);
1134
1135 wxRichTextObjectList::compatibility_iterator objectNode = firstPara->GetChildren().GetFirst();
1136 while (objectNode)
1137 {
1138 wxRichTextObject* newObj = objectNode->GetData()->Clone();
1139
1140 if (!nextObject)
1141 {
1142 // Append
1143 para->AppendChild(newObj);
1144 }
1145 else
1146 {
1147 // Insert before nextObject
1148 para->InsertChild(newObj, nextObject);
1149 }
1150
1151 objectNode = objectNode->GetNext();
1152 }
1153
1154 return true;
1155 }
1156 else
1157 {
1158 // Procedure for inserting a fragment consisting of a number of
1159 // paragraphs:
1160 //
1161 // 1. Remove and save the content that's after the insertion point, for adding
1162 // back once we've added the fragment.
1163 // 2. Add the content from the first fragment paragraph to the current
1164 // paragraph.
1165 // 3. Add remaining fragment paragraphs after the current paragraph.
1166 // 4. Add back the saved content from the first paragraph. If partialParagraph
1167 // is true, add it to the last paragraph added and not a new one.
1168
1169 // 1. Remove and save objects after split point.
1170 wxList savedObjects;
1171 if (nextObject)
1172 para->MoveToList(nextObject, savedObjects);
1173
1174 // 2. Add the content from the 1st fragment paragraph.
1175 wxRichTextObjectList::compatibility_iterator firstParaNode = fragment.GetChildren().GetFirst();
1176 if (!firstParaNode)
1177 return false;
1178
1179 wxRichTextParagraph* firstPara = wxDynamicCast(firstParaNode->GetData(), wxRichTextParagraph);
1180 wxASSERT(firstPara != NULL);
1181
1182 if (!(fragment.GetAttributes().GetFlags() & wxTEXT_ATTR_KEEP_FIRST_PARA_STYLE))
1183 para->SetAttributes(firstPara->GetAttributes());
1184
1185 // Save empty paragraph attributes for appending later
1186 // These are character attributes deliberately set for a new paragraph. Without this,
1187 // we couldn't pass default attributes when appending a new paragraph.
1188 wxTextAttrEx emptyParagraphAttributes;
1189
1190 wxRichTextObjectList::compatibility_iterator objectNode = firstPara->GetChildren().GetFirst();
1191
1192 if (objectNode && firstPara->GetChildren().GetCount() == 1 && objectNode->GetData()->IsEmpty())
1193 emptyParagraphAttributes = objectNode->GetData()->GetAttributes();
1194
1195 while (objectNode)
1196 {
1197 wxRichTextObject* newObj = objectNode->GetData()->Clone();
1198
1199 // Append
1200 para->AppendChild(newObj);
1201
1202 objectNode = objectNode->GetNext();
1203 }
1204
1205 // 3. Add remaining fragment paragraphs after the current paragraph.
1206 wxRichTextObjectList::compatibility_iterator nextParagraphNode = node->GetNext();
1207 wxRichTextObject* nextParagraph = NULL;
1208 if (nextParagraphNode)
1209 nextParagraph = nextParagraphNode->GetData();
1210
1211 wxRichTextObjectList::compatibility_iterator i = fragment.GetChildren().GetFirst()->GetNext();
1212 wxRichTextParagraph* finalPara = para;
1213
1214 bool needExtraPara = (!i || !fragment.GetPartialParagraph());
1215
1216 // If there was only one paragraph, we need to insert a new one.
1217 while (i)
1218 {
1219 wxRichTextParagraph* para = wxDynamicCast(i->GetData(), wxRichTextParagraph);
1220 wxASSERT( para != NULL );
1221
1222 finalPara = (wxRichTextParagraph*) para->Clone();
1223
1224 if (nextParagraph)
1225 InsertChild(finalPara, nextParagraph);
1226 else
1227 AppendChild(finalPara);
1228
1229 i = i->GetNext();
1230 }
1231
1232 // If there was only one paragraph, or we have full paragraphs in our fragment,
1233 // we need to insert a new one.
1234 if (needExtraPara)
1235 {
1236 finalPara = new wxRichTextParagraph;
1237
1238 if (nextParagraph)
1239 InsertChild(finalPara, nextParagraph);
1240 else
1241 AppendChild(finalPara);
1242 }
1243
1244 // 4. Add back the remaining content.
1245 if (finalPara)
1246 {
1247 if (nextObject)
1248 finalPara->MoveFromList(savedObjects);
1249
1250 // Ensure there's at least one object
1251 if (finalPara->GetChildCount() == 0)
1252 {
1253 wxRichTextPlainText* text = new wxRichTextPlainText(wxEmptyString);
1254 text->SetAttributes(emptyParagraphAttributes);
1255
1256 finalPara->AppendChild(text);
1257 }
1258 }
1259
1260 if ((fragment.GetAttributes().GetFlags() & wxTEXT_ATTR_KEEP_FIRST_PARA_STYLE) && firstPara)
1261 finalPara->SetAttributes(firstPara->GetAttributes());
1262 else if (finalPara && finalPara != para)
1263 finalPara->SetAttributes(originalAttr);
1264
1265 return true;
1266 }
1267 }
1268 else
1269 {
1270 // Append
1271 wxRichTextObjectList::compatibility_iterator i = fragment.GetChildren().GetFirst();
1272 while (i)
1273 {
1274 wxRichTextParagraph* para = wxDynamicCast(i->GetData(), wxRichTextParagraph);
1275 wxASSERT( para != NULL );
1276
1277 AppendChild(para->Clone());
1278
1279 i = i->GetNext();
1280 }
1281
1282 return true;
1283 }
1284 }
1285
1286 /// Make a copy of the fragment corresponding to the given range, putting it in 'fragment'.
1287 /// If there was an incomplete paragraph at the end, partialParagraph is set to true.
1288 bool wxRichTextParagraphLayoutBox::CopyFragment(const wxRichTextRange& range, wxRichTextParagraphLayoutBox& fragment)
1289 {
1290 wxRichTextObjectList::compatibility_iterator i = GetChildren().GetFirst();
1291 while (i)
1292 {
1293 wxRichTextParagraph* para = wxDynamicCast(i->GetData(), wxRichTextParagraph);
1294 wxASSERT( para != NULL );
1295
1296 if (!para->GetRange().IsOutside(range))
1297 {
1298 fragment.AppendChild(para->Clone());
1299 }
1300 i = i->GetNext();
1301 }
1302
1303 // Now top and tail the first and last paragraphs in our new fragment (which might be the same).
1304 if (!fragment.IsEmpty())
1305 {
1306 wxRichTextRange topTailRange(range);
1307
1308 wxRichTextParagraph* firstPara = wxDynamicCast(fragment.GetChildren().GetFirst()->GetData(), wxRichTextParagraph);
1309 wxASSERT( firstPara != NULL );
1310
1311 // Chop off the start of the paragraph
1312 if (topTailRange.GetStart() > firstPara->GetRange().GetStart())
1313 {
1314 wxRichTextRange r(firstPara->GetRange().GetStart(), topTailRange.GetStart()-1);
1315 firstPara->DeleteRange(r);
1316
1317 // Make sure the numbering is correct
1318 long end;
1319 fragment.CalculateRange(firstPara->GetRange().GetStart(), end);
1320
1321 // Now, we've deleted some positions, so adjust the range
1322 // accordingly.
1323 topTailRange.SetEnd(topTailRange.GetEnd() - r.GetLength());
1324 }
1325
1326 wxRichTextParagraph* lastPara = wxDynamicCast(fragment.GetChildren().GetLast()->GetData(), wxRichTextParagraph);
1327 wxASSERT( lastPara != NULL );
1328
1329 if (topTailRange.GetEnd() < (lastPara->GetRange().GetEnd()-1))
1330 {
1331 wxRichTextRange r(topTailRange.GetEnd()+1, lastPara->GetRange().GetEnd()-1); /* -1 since actual text ends 1 position before end of para marker */
1332 lastPara->DeleteRange(r);
1333
1334 // Make sure the numbering is correct
1335 long end;
1336 fragment.CalculateRange(firstPara->GetRange().GetStart(), end);
1337
1338 // We only have part of a paragraph at the end
1339 fragment.SetPartialParagraph(true);
1340 }
1341 else
1342 {
1343 if (topTailRange.GetEnd() == (lastPara->GetRange().GetEnd() - 1))
1344 // We have a partial paragraph (don't save last new paragraph marker)
1345 fragment.SetPartialParagraph(true);
1346 else
1347 // We have a complete paragraph
1348 fragment.SetPartialParagraph(false);
1349 }
1350 }
1351
1352 return true;
1353 }
1354
1355 /// Given a position, get the number of the visible line (potentially many to a paragraph),
1356 /// starting from zero at the start of the buffer.
1357 long wxRichTextParagraphLayoutBox::GetVisibleLineNumber(long pos, bool caretPosition, bool startOfLine) const
1358 {
1359 if (caretPosition)
1360 pos ++;
1361
1362 int lineCount = 0;
1363
1364 wxRichTextObjectList::compatibility_iterator node = m_children.GetFirst();
1365 while (node)
1366 {
1367 wxRichTextParagraph* child = wxDynamicCast(node->GetData(), wxRichTextParagraph);
1368 wxASSERT( child != NULL );
1369
1370 if (child->GetRange().Contains(pos))
1371 {
1372 wxRichTextLineList::compatibility_iterator node2 = child->GetLines().GetFirst();
1373 while (node2)
1374 {
1375 wxRichTextLine* line = node2->GetData();
1376 wxRichTextRange lineRange = line->GetAbsoluteRange();
1377
1378 if (lineRange.Contains(pos))
1379 {
1380 // If the caret is displayed at the end of the previous wrapped line,
1381 // we want to return the line it's _displayed_ at (not the actual line
1382 // containing the position).
1383 if (lineRange.GetStart() == pos && !startOfLine && child->GetRange().GetStart() != pos)
1384 return lineCount - 1;
1385 else
1386 return lineCount;
1387 }
1388
1389 lineCount ++;
1390
1391 node2 = node2->GetNext();
1392 }
1393 // If we didn't find it in the lines, it must be
1394 // the last position of the paragraph. So return the last line.
1395 return lineCount-1;
1396 }
1397 else
1398 lineCount += child->GetLines().GetCount();
1399
1400 node = node->GetNext();
1401 }
1402
1403 // Not found
1404 return -1;
1405 }
1406
1407 /// Given a line number, get the corresponding wxRichTextLine object.
1408 wxRichTextLine* wxRichTextParagraphLayoutBox::GetLineForVisibleLineNumber(long lineNumber) const
1409 {
1410 int lineCount = 0;
1411
1412 wxRichTextObjectList::compatibility_iterator node = m_children.GetFirst();
1413 while (node)
1414 {
1415 wxRichTextParagraph* child = wxDynamicCast(node->GetData(), wxRichTextParagraph);
1416 wxASSERT(child != NULL);
1417
1418 if (lineNumber < (int) (child->GetLines().GetCount() + lineCount))
1419 {
1420 wxRichTextLineList::compatibility_iterator node2 = child->GetLines().GetFirst();
1421 while (node2)
1422 {
1423 wxRichTextLine* line = node2->GetData();
1424
1425 if (lineCount == lineNumber)
1426 return line;
1427
1428 lineCount ++;
1429
1430 node2 = node2->GetNext();
1431 }
1432 }
1433 else
1434 lineCount += child->GetLines().GetCount();
1435
1436 node = node->GetNext();
1437 }
1438
1439 // Didn't find it
1440 return NULL;
1441 }
1442
1443 /// Delete range from layout.
1444 bool wxRichTextParagraphLayoutBox::DeleteRange(const wxRichTextRange& range)
1445 {
1446 wxRichTextObjectList::compatibility_iterator node = m_children.GetFirst();
1447
1448 wxRichTextParagraph* firstPara = NULL;
1449 while (node)
1450 {
1451 wxRichTextParagraph* obj = wxDynamicCast(node->GetData(), wxRichTextParagraph);
1452 wxASSERT (obj != NULL);
1453
1454 wxRichTextObjectList::compatibility_iterator next = node->GetNext();
1455
1456 // Delete the range in each paragraph
1457
1458 if (!obj->GetRange().IsOutside(range))
1459 {
1460 // Deletes the content of this object within the given range
1461 obj->DeleteRange(range);
1462
1463 wxRichTextRange thisRange = obj->GetRange();
1464 wxTextAttrEx thisAttr = obj->GetAttributes();
1465
1466 // If the whole paragraph is within the range to delete,
1467 // delete the whole thing.
1468 if (range.GetStart() <= thisRange.GetStart() && range.GetEnd() >= thisRange.GetEnd())
1469 {
1470 // Delete the whole object
1471 RemoveChild(obj, true);
1472 obj = NULL;
1473 }
1474 else if (!firstPara)
1475 firstPara = obj;
1476
1477 // If the range includes the paragraph end, we need to join this
1478 // and the next paragraph.
1479 if (range.GetEnd() <= thisRange.GetEnd())
1480 {
1481 // We need to move the objects from the next paragraph
1482 // to this paragraph
1483
1484 wxRichTextParagraph* nextParagraph = NULL;
1485 if ((range.GetEnd() < thisRange.GetEnd()) && obj)
1486 nextParagraph = obj;
1487 else
1488 {
1489 // We're ending at the end of the paragraph, so merge the _next_ paragraph.
1490 if (next)
1491 nextParagraph = wxDynamicCast(next->GetData(), wxRichTextParagraph);
1492 }
1493
1494 bool applyFinalParagraphStyle = firstPara && nextParagraph && nextParagraph != firstPara;
1495
1496 wxTextAttrEx nextParaAttr;
1497 if (applyFinalParagraphStyle)
1498 {
1499 // Special case when deleting the end of a paragraph - use _this_ paragraph's style,
1500 // not the next one.
1501 if (range.GetStart() == range.GetEnd() && range.GetStart() == thisRange.GetEnd())
1502 nextParaAttr = thisAttr;
1503 else
1504 nextParaAttr = nextParagraph->GetAttributes();
1505 }
1506
1507 if (firstPara && nextParagraph && firstPara != nextParagraph)
1508 {
1509 // Move the objects to the previous para
1510 wxRichTextObjectList::compatibility_iterator node1 = nextParagraph->GetChildren().GetFirst();
1511
1512 while (node1)
1513 {
1514 wxRichTextObject* obj1 = node1->GetData();
1515
1516 firstPara->AppendChild(obj1);
1517
1518 wxRichTextObjectList::compatibility_iterator next1 = node1->GetNext();
1519 nextParagraph->GetChildren().Erase(node1);
1520
1521 node1 = next1;
1522 }
1523
1524 // Delete the paragraph
1525 RemoveChild(nextParagraph, true);
1526 }
1527
1528 // Avoid empty paragraphs
1529 if (firstPara && firstPara->GetChildren().GetCount() == 0)
1530 {
1531 wxRichTextPlainText* text = new wxRichTextPlainText(wxEmptyString);
1532 firstPara->AppendChild(text);
1533 }
1534
1535 if (applyFinalParagraphStyle)
1536 firstPara->SetAttributes(nextParaAttr);
1537
1538 return true;
1539 }
1540 }
1541
1542 node = next;
1543 }
1544
1545 return true;
1546 }
1547
1548 /// Get any text in this object for the given range
1549 wxString wxRichTextParagraphLayoutBox::GetTextForRange(const wxRichTextRange& range) const
1550 {
1551 int lineCount = 0;
1552 wxString text;
1553 wxRichTextObjectList::compatibility_iterator node = m_children.GetFirst();
1554 while (node)
1555 {
1556 wxRichTextObject* child = node->GetData();
1557 if (!child->GetRange().IsOutside(range))
1558 {
1559 wxRichTextRange childRange = range;
1560 childRange.LimitTo(child->GetRange());
1561
1562 wxString childText = child->GetTextForRange(childRange);
1563
1564 text += childText;
1565
1566 if ((childRange.GetEnd() == child->GetRange().GetEnd()) && node->GetNext())
1567 text += wxT("\n");
1568
1569 lineCount ++;
1570 }
1571 node = node->GetNext();
1572 }
1573
1574 return text;
1575 }
1576
1577 /// Get all the text
1578 wxString wxRichTextParagraphLayoutBox::GetText() const
1579 {
1580 return GetTextForRange(GetRange());
1581 }
1582
1583 /// Get the paragraph by number
1584 wxRichTextParagraph* wxRichTextParagraphLayoutBox::GetParagraphAtLine(long paragraphNumber) const
1585 {
1586 if ((size_t) paragraphNumber >= GetChildCount())
1587 return NULL;
1588
1589 return (wxRichTextParagraph*) GetChild((size_t) paragraphNumber);
1590 }
1591
1592 /// Get the length of the paragraph
1593 int wxRichTextParagraphLayoutBox::GetParagraphLength(long paragraphNumber) const
1594 {
1595 wxRichTextParagraph* para = GetParagraphAtLine(paragraphNumber);
1596 if (para)
1597 return para->GetRange().GetLength() - 1; // don't include newline
1598 else
1599 return 0;
1600 }
1601
1602 /// Get the text of the paragraph
1603 wxString wxRichTextParagraphLayoutBox::GetParagraphText(long paragraphNumber) const
1604 {
1605 wxRichTextParagraph* para = GetParagraphAtLine(paragraphNumber);
1606 if (para)
1607 return para->GetTextForRange(para->GetRange());
1608 else
1609 return wxEmptyString;
1610 }
1611
1612 /// Convert zero-based line column and paragraph number to a position.
1613 long wxRichTextParagraphLayoutBox::XYToPosition(long x, long y) const
1614 {
1615 wxRichTextParagraph* para = GetParagraphAtLine(y);
1616 if (para)
1617 {
1618 return para->GetRange().GetStart() + x;
1619 }
1620 else
1621 return -1;
1622 }
1623
1624 /// Convert zero-based position to line column and paragraph number
1625 bool wxRichTextParagraphLayoutBox::PositionToXY(long pos, long* x, long* y) const
1626 {
1627 wxRichTextParagraph* para = GetParagraphAtPosition(pos);
1628 if (para)
1629 {
1630 int count = 0;
1631 wxRichTextObjectList::compatibility_iterator node = m_children.GetFirst();
1632 while (node)
1633 {
1634 wxRichTextObject* child = node->GetData();
1635 if (child == para)
1636 break;
1637 count ++;
1638 node = node->GetNext();
1639 }
1640
1641 *y = count;
1642 *x = pos - para->GetRange().GetStart();
1643
1644 return true;
1645 }
1646 else
1647 return false;
1648 }
1649
1650 /// Get the leaf object in a paragraph at this position.
1651 /// Given a line number, get the corresponding wxRichTextLine object.
1652 wxRichTextObject* wxRichTextParagraphLayoutBox::GetLeafObjectAtPosition(long position) const
1653 {
1654 wxRichTextParagraph* para = GetParagraphAtPosition(position);
1655 if (para)
1656 {
1657 wxRichTextObjectList::compatibility_iterator node = para->GetChildren().GetFirst();
1658
1659 while (node)
1660 {
1661 wxRichTextObject* child = node->GetData();
1662 if (child->GetRange().Contains(position))
1663 return child;
1664
1665 node = node->GetNext();
1666 }
1667 if (position == para->GetRange().GetEnd() && para->GetChildCount() > 0)
1668 return para->GetChildren().GetLast()->GetData();
1669 }
1670 return NULL;
1671 }
1672
1673 /// Set character or paragraph text attributes: apply character styles only to immediate text nodes
1674 bool wxRichTextParagraphLayoutBox::SetStyle(const wxRichTextRange& range, const wxTextAttr& style, int flags)
1675 {
1676 bool characterStyle = false;
1677 bool paragraphStyle = false;
1678
1679 if (style.IsCharacterStyle())
1680 characterStyle = true;
1681 if (style.IsParagraphStyle())
1682 paragraphStyle = true;
1683
1684 bool withUndo = ((flags & wxRICHTEXT_SETSTYLE_WITH_UNDO) != 0);
1685 bool applyMinimal = ((flags & wxRICHTEXT_SETSTYLE_OPTIMIZE) != 0);
1686 bool parasOnly = ((flags & wxRICHTEXT_SETSTYLE_PARAGRAPHS_ONLY) != 0);
1687 bool charactersOnly = ((flags & wxRICHTEXT_SETSTYLE_CHARACTERS_ONLY) != 0);
1688 bool resetExistingStyle = ((flags & wxRICHTEXT_SETSTYLE_RESET) != 0);
1689 bool removeStyle = ((flags & wxRICHTEXT_SETSTYLE_REMOVE) != 0);
1690
1691 // Apply paragraph style first, if any
1692 wxTextAttr wholeStyle(style);
1693
1694 if (!removeStyle && wholeStyle.HasParagraphStyleName() && GetStyleSheet())
1695 {
1696 wxRichTextParagraphStyleDefinition* def = GetStyleSheet()->FindParagraphStyle(wholeStyle.GetParagraphStyleName());
1697 if (def)
1698 wxRichTextApplyStyle(wholeStyle, def->GetStyleMergedWithBase(GetStyleSheet()));
1699 }
1700
1701 // Limit the attributes to be set to the content to only character attributes.
1702 wxTextAttr characterAttributes(wholeStyle);
1703 characterAttributes.SetFlags(characterAttributes.GetFlags() & (wxTEXT_ATTR_CHARACTER));
1704
1705 if (!removeStyle && characterAttributes.HasCharacterStyleName() && GetStyleSheet())
1706 {
1707 wxRichTextCharacterStyleDefinition* def = GetStyleSheet()->FindCharacterStyle(characterAttributes.GetCharacterStyleName());
1708 if (def)
1709 wxRichTextApplyStyle(characterAttributes, def->GetStyleMergedWithBase(GetStyleSheet()));
1710 }
1711
1712 // If we are associated with a control, make undoable; otherwise, apply immediately
1713 // to the data.
1714
1715 bool haveControl = (GetRichTextCtrl() != NULL);
1716
1717 wxRichTextAction* action = NULL;
1718
1719 if (haveControl && withUndo)
1720 {
1721 action = new wxRichTextAction(NULL, _("Change Style"), wxRICHTEXT_CHANGE_STYLE, & GetRichTextCtrl()->GetBuffer(), GetRichTextCtrl());
1722 action->SetRange(range);
1723 action->SetPosition(GetRichTextCtrl()->GetCaretPosition());
1724 }
1725
1726 wxRichTextObjectList::compatibility_iterator node = m_children.GetFirst();
1727 while (node)
1728 {
1729 wxRichTextParagraph* para = wxDynamicCast(node->GetData(), wxRichTextParagraph);
1730 wxASSERT (para != NULL);
1731
1732 if (para && para->GetChildCount() > 0)
1733 {
1734 // Stop searching if we're beyond the range of interest
1735 if (para->GetRange().GetStart() > range.GetEnd())
1736 break;
1737
1738 if (!para->GetRange().IsOutside(range))
1739 {
1740 // We'll be using a copy of the paragraph to make style changes,
1741 // not updating the buffer directly.
1742 wxRichTextParagraph* newPara wxDUMMY_INITIALIZE(NULL);
1743
1744 if (haveControl && withUndo)
1745 {
1746 newPara = new wxRichTextParagraph(*para);
1747 action->GetNewParagraphs().AppendChild(newPara);
1748
1749 // Also store the old ones for Undo
1750 action->GetOldParagraphs().AppendChild(new wxRichTextParagraph(*para));
1751 }
1752 else
1753 newPara = para;
1754
1755 // If we're specifying paragraphs only, then we really mean character formatting
1756 // to be included in the paragraph style
1757 if ((paragraphStyle || parasOnly) && !charactersOnly)
1758 {
1759 if (removeStyle)
1760 {
1761 // Removes the given style from the paragraph
1762 wxRichTextRemoveStyle(newPara->GetAttributes(), style);
1763 }
1764 else if (resetExistingStyle)
1765 newPara->GetAttributes() = wholeStyle;
1766 else
1767 {
1768 if (applyMinimal)
1769 {
1770 // Only apply attributes that will make a difference to the combined
1771 // style as seen on the display
1772 wxTextAttr combinedAttr(para->GetCombinedAttributes());
1773 wxRichTextApplyStyle(newPara->GetAttributes(), wholeStyle, & combinedAttr);
1774 }
1775 else
1776 wxRichTextApplyStyle(newPara->GetAttributes(), wholeStyle);
1777 }
1778 }
1779
1780 // When applying paragraph styles dynamically, don't change the text objects' attributes
1781 // since they will computed as needed. Only apply the character styling if it's _only_
1782 // character styling. This policy is subject to change and might be put under user control.
1783
1784 // Hm. we might well be applying a mix of paragraph and character styles, in which
1785 // case we _do_ want to apply character styles regardless of what para styles are set.
1786 // But if we're applying a paragraph style, which has some character attributes, but
1787 // we only want the paragraphs to hold this character style, then we _don't_ want to
1788 // apply the character style. So we need to be able to choose.
1789
1790 // if (!paragraphStyle && characterStyle && range.GetStart() != newPara->GetRange().GetEnd())
1791 if (!parasOnly && characterStyle && range.GetStart() != newPara->GetRange().GetEnd())
1792 {
1793 wxRichTextRange childRange(range);
1794 childRange.LimitTo(newPara->GetRange());
1795
1796 // Find the starting position and if necessary split it so
1797 // we can start applying a different style.
1798 // TODO: check that the style actually changes or is different
1799 // from style outside of range
1800 wxRichTextObject* firstObject wxDUMMY_INITIALIZE(NULL);
1801 wxRichTextObject* lastObject wxDUMMY_INITIALIZE(NULL);
1802
1803 if (childRange.GetStart() == newPara->GetRange().GetStart())
1804 firstObject = newPara->GetChildren().GetFirst()->GetData();
1805 else
1806 firstObject = newPara->SplitAt(range.GetStart());
1807
1808 // Increment by 1 because we're apply the style one _after_ the split point
1809 long splitPoint = childRange.GetEnd();
1810 if (splitPoint != newPara->GetRange().GetEnd())
1811 splitPoint ++;
1812
1813 // Find last object
1814 if (splitPoint == newPara->GetRange().GetEnd())
1815 lastObject = newPara->GetChildren().GetLast()->GetData();
1816 else
1817 // lastObject is set as a side-effect of splitting. It's
1818 // returned as the object before the new object.
1819 (void) newPara->SplitAt(splitPoint, & lastObject);
1820
1821 wxASSERT(firstObject != NULL);
1822 wxASSERT(lastObject != NULL);
1823
1824 if (!firstObject || !lastObject)
1825 continue;
1826
1827 wxRichTextObjectList::compatibility_iterator firstNode = newPara->GetChildren().Find(firstObject);
1828 wxRichTextObjectList::compatibility_iterator lastNode = newPara->GetChildren().Find(lastObject);
1829
1830 wxASSERT(firstNode);
1831 wxASSERT(lastNode);
1832
1833 wxRichTextObjectList::compatibility_iterator node2 = firstNode;
1834
1835 while (node2)
1836 {
1837 wxRichTextObject* child = node2->GetData();
1838
1839 if (removeStyle)
1840 {
1841 // Removes the given style from the paragraph
1842 wxRichTextRemoveStyle(child->GetAttributes(), style);
1843 }
1844 else if (resetExistingStyle)
1845 child->GetAttributes() = characterAttributes;
1846 else
1847 {
1848 if (applyMinimal)
1849 {
1850 // Only apply attributes that will make a difference to the combined
1851 // style as seen on the display
1852 wxTextAttr combinedAttr(newPara->GetCombinedAttributes(child->GetAttributes()));
1853 wxRichTextApplyStyle(child->GetAttributes(), characterAttributes, & combinedAttr);
1854 }
1855 else
1856 wxRichTextApplyStyle(child->GetAttributes(), characterAttributes);
1857 }
1858
1859 if (node2 == lastNode)
1860 break;
1861
1862 node2 = node2->GetNext();
1863 }
1864 }
1865 }
1866 }
1867
1868 node = node->GetNext();
1869 }
1870
1871 // Do action, or delay it until end of batch.
1872 if (haveControl && withUndo)
1873 GetRichTextCtrl()->GetBuffer().SubmitAction(action);
1874
1875 return true;
1876 }
1877
1878 /// Get the text attributes for this position.
1879 bool wxRichTextParagraphLayoutBox::GetStyle(long position, wxTextAttr& style)
1880 {
1881 return DoGetStyle(position, style, true);
1882 }
1883
1884 bool wxRichTextParagraphLayoutBox::GetUncombinedStyle(long position, wxTextAttr& style)
1885 {
1886 return DoGetStyle(position, style, false);
1887 }
1888
1889 /// Implementation helper for GetStyle. If combineStyles is true, combine base, paragraph and
1890 /// context attributes.
1891 bool wxRichTextParagraphLayoutBox::DoGetStyle(long position, wxTextAttr& style, bool combineStyles)
1892 {
1893 wxRichTextObject* obj wxDUMMY_INITIALIZE(NULL);
1894
1895 if (style.IsParagraphStyle())
1896 {
1897 obj = GetParagraphAtPosition(position);
1898 if (obj)
1899 {
1900 if (combineStyles)
1901 {
1902 // Start with the base style
1903 style = GetAttributes();
1904
1905 // Apply the paragraph style
1906 wxRichTextApplyStyle(style, obj->GetAttributes());
1907 }
1908 else
1909 style = obj->GetAttributes();
1910
1911 return true;
1912 }
1913 }
1914 else
1915 {
1916 obj = GetLeafObjectAtPosition(position);
1917 if (obj)
1918 {
1919 if (combineStyles)
1920 {
1921 wxRichTextParagraph* para = wxDynamicCast(obj->GetParent(), wxRichTextParagraph);
1922 style = para ? para->GetCombinedAttributes(obj->GetAttributes()) : obj->GetAttributes();
1923 }
1924 else
1925 style = obj->GetAttributes();
1926
1927 return true;
1928 }
1929 }
1930 return false;
1931 }
1932
1933 static bool wxHasStyle(long flags, long style)
1934 {
1935 return (flags & style) != 0;
1936 }
1937
1938 /// Combines 'style' with 'currentStyle' for the purpose of summarising the attributes of a range of
1939 /// content.
1940 bool wxRichTextParagraphLayoutBox::CollectStyle(wxTextAttr& currentStyle, const wxTextAttr& style, long& multipleStyleAttributes, int& multipleTextEffectAttributes)
1941 {
1942 if (style.HasFont())
1943 {
1944 if (style.HasFontSize() && !wxHasStyle(multipleStyleAttributes, wxTEXT_ATTR_FONT_SIZE))
1945 {
1946 if (currentStyle.HasFontSize())
1947 {
1948 if (currentStyle.GetFontSize() != style.GetFontSize())
1949 {
1950 // Clash of style - mark as such
1951 multipleStyleAttributes |= wxTEXT_ATTR_FONT_SIZE;
1952 currentStyle.SetFlags(currentStyle.GetFlags() & ~wxTEXT_ATTR_FONT_SIZE);
1953 }
1954 }
1955 else
1956 {
1957 currentStyle.SetFontSize(style.GetFontSize());
1958 }
1959 }
1960
1961 if (style.HasFontItalic() && !wxHasStyle(multipleStyleAttributes, wxTEXT_ATTR_FONT_ITALIC))
1962 {
1963 if (currentStyle.HasFontItalic())
1964 {
1965 if (currentStyle.GetFontStyle() != style.GetFontStyle())
1966 {
1967 // Clash of style - mark as such
1968 multipleStyleAttributes |= wxTEXT_ATTR_FONT_ITALIC;
1969 currentStyle.SetFlags(currentStyle.GetFlags() & ~wxTEXT_ATTR_FONT_ITALIC);
1970 }
1971 }
1972 else
1973 {
1974 currentStyle.SetFontStyle(style.GetFontStyle());
1975 }
1976 }
1977
1978 if (style.HasFontWeight() && !wxHasStyle(multipleStyleAttributes, wxTEXT_ATTR_FONT_WEIGHT))
1979 {
1980 if (currentStyle.HasFontWeight())
1981 {
1982 if (currentStyle.GetFontWeight() != style.GetFontWeight())
1983 {
1984 // Clash of style - mark as such
1985 multipleStyleAttributes |= wxTEXT_ATTR_FONT_WEIGHT;
1986 currentStyle.SetFlags(currentStyle.GetFlags() & ~wxTEXT_ATTR_FONT_WEIGHT);
1987 }
1988 }
1989 else
1990 {
1991 currentStyle.SetFontWeight(style.GetFontWeight());
1992 }
1993 }
1994
1995 if (style.HasFontFaceName() && !wxHasStyle(multipleStyleAttributes, wxTEXT_ATTR_FONT_FACE))
1996 {
1997 if (currentStyle.HasFontFaceName())
1998 {
1999 wxString faceName1(currentStyle.GetFontFaceName());
2000 wxString faceName2(style.GetFontFaceName());
2001
2002 if (faceName1 != faceName2)
2003 {
2004 // Clash of style - mark as such
2005 multipleStyleAttributes |= wxTEXT_ATTR_FONT_FACE;
2006 currentStyle.SetFlags(currentStyle.GetFlags() & ~wxTEXT_ATTR_FONT_FACE);
2007 }
2008 }
2009 else
2010 {
2011 currentStyle.SetFontFaceName(style.GetFontFaceName());
2012 }
2013 }
2014
2015 if (style.HasFontUnderlined() && !wxHasStyle(multipleStyleAttributes, wxTEXT_ATTR_FONT_UNDERLINE))
2016 {
2017 if (currentStyle.HasFontUnderlined())
2018 {
2019 if (currentStyle.GetFontUnderlined() != style.GetFontUnderlined())
2020 {
2021 // Clash of style - mark as such
2022 multipleStyleAttributes |= wxTEXT_ATTR_FONT_UNDERLINE;
2023 currentStyle.SetFlags(currentStyle.GetFlags() & ~wxTEXT_ATTR_FONT_UNDERLINE);
2024 }
2025 }
2026 else
2027 {
2028 currentStyle.SetFontUnderlined(style.GetFontUnderlined());
2029 }
2030 }
2031 }
2032
2033 if (style.HasTextColour() && !wxHasStyle(multipleStyleAttributes, wxTEXT_ATTR_TEXT_COLOUR))
2034 {
2035 if (currentStyle.HasTextColour())
2036 {
2037 if (currentStyle.GetTextColour() != style.GetTextColour())
2038 {
2039 // Clash of style - mark as such
2040 multipleStyleAttributes |= wxTEXT_ATTR_TEXT_COLOUR;
2041 currentStyle.SetFlags(currentStyle.GetFlags() & ~wxTEXT_ATTR_TEXT_COLOUR);
2042 }
2043 }
2044 else
2045 currentStyle.SetTextColour(style.GetTextColour());
2046 }
2047
2048 if (style.HasBackgroundColour() && !wxHasStyle(multipleStyleAttributes, wxTEXT_ATTR_BACKGROUND_COLOUR))
2049 {
2050 if (currentStyle.HasBackgroundColour())
2051 {
2052 if (currentStyle.GetBackgroundColour() != style.GetBackgroundColour())
2053 {
2054 // Clash of style - mark as such
2055 multipleStyleAttributes |= wxTEXT_ATTR_BACKGROUND_COLOUR;
2056 currentStyle.SetFlags(currentStyle.GetFlags() & ~wxTEXT_ATTR_BACKGROUND_COLOUR);
2057 }
2058 }
2059 else
2060 currentStyle.SetBackgroundColour(style.GetBackgroundColour());
2061 }
2062
2063 if (style.HasAlignment() && !wxHasStyle(multipleStyleAttributes, wxTEXT_ATTR_ALIGNMENT))
2064 {
2065 if (currentStyle.HasAlignment())
2066 {
2067 if (currentStyle.GetAlignment() != style.GetAlignment())
2068 {
2069 // Clash of style - mark as such
2070 multipleStyleAttributes |= wxTEXT_ATTR_ALIGNMENT;
2071 currentStyle.SetFlags(currentStyle.GetFlags() & ~wxTEXT_ATTR_ALIGNMENT);
2072 }
2073 }
2074 else
2075 currentStyle.SetAlignment(style.GetAlignment());
2076 }
2077
2078 if (style.HasTabs() && !wxHasStyle(multipleStyleAttributes, wxTEXT_ATTR_TABS))
2079 {
2080 if (currentStyle.HasTabs())
2081 {
2082 if (!wxRichTextTabsEq(currentStyle.GetTabs(), style.GetTabs()))
2083 {
2084 // Clash of style - mark as such
2085 multipleStyleAttributes |= wxTEXT_ATTR_TABS;
2086 currentStyle.SetFlags(currentStyle.GetFlags() & ~wxTEXT_ATTR_TABS);
2087 }
2088 }
2089 else
2090 currentStyle.SetTabs(style.GetTabs());
2091 }
2092
2093 if (style.HasLeftIndent() && !wxHasStyle(multipleStyleAttributes, wxTEXT_ATTR_LEFT_INDENT))
2094 {
2095 if (currentStyle.HasLeftIndent())
2096 {
2097 if (currentStyle.GetLeftIndent() != style.GetLeftIndent() || currentStyle.GetLeftSubIndent() != style.GetLeftSubIndent())
2098 {
2099 // Clash of style - mark as such
2100 multipleStyleAttributes |= wxTEXT_ATTR_LEFT_INDENT;
2101 currentStyle.SetFlags(currentStyle.GetFlags() & ~wxTEXT_ATTR_LEFT_INDENT);
2102 }
2103 }
2104 else
2105 currentStyle.SetLeftIndent(style.GetLeftIndent(), style.GetLeftSubIndent());
2106 }
2107
2108 if (style.HasRightIndent() && !wxHasStyle(multipleStyleAttributes, wxTEXT_ATTR_RIGHT_INDENT))
2109 {
2110 if (currentStyle.HasRightIndent())
2111 {
2112 if (currentStyle.GetRightIndent() != style.GetRightIndent())
2113 {
2114 // Clash of style - mark as such
2115 multipleStyleAttributes |= wxTEXT_ATTR_RIGHT_INDENT;
2116 currentStyle.SetFlags(currentStyle.GetFlags() & ~wxTEXT_ATTR_RIGHT_INDENT);
2117 }
2118 }
2119 else
2120 currentStyle.SetRightIndent(style.GetRightIndent());
2121 }
2122
2123 if (style.HasParagraphSpacingAfter() && !wxHasStyle(multipleStyleAttributes, wxTEXT_ATTR_PARA_SPACING_AFTER))
2124 {
2125 if (currentStyle.HasParagraphSpacingAfter())
2126 {
2127 if (currentStyle.GetParagraphSpacingAfter() != style.GetParagraphSpacingAfter())
2128 {
2129 // Clash of style - mark as such
2130 multipleStyleAttributes |= wxTEXT_ATTR_PARA_SPACING_AFTER;
2131 currentStyle.SetFlags(currentStyle.GetFlags() & ~wxTEXT_ATTR_PARA_SPACING_AFTER);
2132 }
2133 }
2134 else
2135 currentStyle.SetParagraphSpacingAfter(style.GetParagraphSpacingAfter());
2136 }
2137
2138 if (style.HasParagraphSpacingBefore() && !wxHasStyle(multipleStyleAttributes, wxTEXT_ATTR_PARA_SPACING_BEFORE))
2139 {
2140 if (currentStyle.HasParagraphSpacingBefore())
2141 {
2142 if (currentStyle.GetParagraphSpacingBefore() != style.GetParagraphSpacingBefore())
2143 {
2144 // Clash of style - mark as such
2145 multipleStyleAttributes |= wxTEXT_ATTR_PARA_SPACING_BEFORE;
2146 currentStyle.SetFlags(currentStyle.GetFlags() & ~wxTEXT_ATTR_PARA_SPACING_BEFORE);
2147 }
2148 }
2149 else
2150 currentStyle.SetParagraphSpacingBefore(style.GetParagraphSpacingBefore());
2151 }
2152
2153 if (style.HasLineSpacing() && !wxHasStyle(multipleStyleAttributes, wxTEXT_ATTR_LINE_SPACING))
2154 {
2155 if (currentStyle.HasLineSpacing())
2156 {
2157 if (currentStyle.GetLineSpacing() != style.GetLineSpacing())
2158 {
2159 // Clash of style - mark as such
2160 multipleStyleAttributes |= wxTEXT_ATTR_LINE_SPACING;
2161 currentStyle.SetFlags(currentStyle.GetFlags() & ~wxTEXT_ATTR_LINE_SPACING);
2162 }
2163 }
2164 else
2165 currentStyle.SetLineSpacing(style.GetLineSpacing());
2166 }
2167
2168 if (style.HasCharacterStyleName() && !wxHasStyle(multipleStyleAttributes, wxTEXT_ATTR_CHARACTER_STYLE_NAME))
2169 {
2170 if (currentStyle.HasCharacterStyleName())
2171 {
2172 if (currentStyle.GetCharacterStyleName() != style.GetCharacterStyleName())
2173 {
2174 // Clash of style - mark as such
2175 multipleStyleAttributes |= wxTEXT_ATTR_CHARACTER_STYLE_NAME;
2176 currentStyle.SetFlags(currentStyle.GetFlags() & ~wxTEXT_ATTR_CHARACTER_STYLE_NAME);
2177 }
2178 }
2179 else
2180 currentStyle.SetCharacterStyleName(style.GetCharacterStyleName());
2181 }
2182
2183 if (style.HasParagraphStyleName() && !wxHasStyle(multipleStyleAttributes, wxTEXT_ATTR_PARAGRAPH_STYLE_NAME))
2184 {
2185 if (currentStyle.HasParagraphStyleName())
2186 {
2187 if (currentStyle.GetParagraphStyleName() != style.GetParagraphStyleName())
2188 {
2189 // Clash of style - mark as such
2190 multipleStyleAttributes |= wxTEXT_ATTR_PARAGRAPH_STYLE_NAME;
2191 currentStyle.SetFlags(currentStyle.GetFlags() & ~wxTEXT_ATTR_PARAGRAPH_STYLE_NAME);
2192 }
2193 }
2194 else
2195 currentStyle.SetParagraphStyleName(style.GetParagraphStyleName());
2196 }
2197
2198 if (style.HasListStyleName() && !wxHasStyle(multipleStyleAttributes, wxTEXT_ATTR_LIST_STYLE_NAME))
2199 {
2200 if (currentStyle.HasListStyleName())
2201 {
2202 if (currentStyle.GetListStyleName() != style.GetListStyleName())
2203 {
2204 // Clash of style - mark as such
2205 multipleStyleAttributes |= wxTEXT_ATTR_LIST_STYLE_NAME;
2206 currentStyle.SetFlags(currentStyle.GetFlags() & ~wxTEXT_ATTR_LIST_STYLE_NAME);
2207 }
2208 }
2209 else
2210 currentStyle.SetListStyleName(style.GetListStyleName());
2211 }
2212
2213 if (style.HasBulletStyle() && !wxHasStyle(multipleStyleAttributes, wxTEXT_ATTR_BULLET_STYLE))
2214 {
2215 if (currentStyle.HasBulletStyle())
2216 {
2217 if (currentStyle.GetBulletStyle() != style.GetBulletStyle())
2218 {
2219 // Clash of style - mark as such
2220 multipleStyleAttributes |= wxTEXT_ATTR_BULLET_STYLE;
2221 currentStyle.SetFlags(currentStyle.GetFlags() & ~wxTEXT_ATTR_BULLET_STYLE);
2222 }
2223 }
2224 else
2225 currentStyle.SetBulletStyle(style.GetBulletStyle());
2226 }
2227
2228 if (style.HasBulletNumber() && !wxHasStyle(multipleStyleAttributes, wxTEXT_ATTR_BULLET_NUMBER))
2229 {
2230 if (currentStyle.HasBulletNumber())
2231 {
2232 if (currentStyle.GetBulletNumber() != style.GetBulletNumber())
2233 {
2234 // Clash of style - mark as such
2235 multipleStyleAttributes |= wxTEXT_ATTR_BULLET_NUMBER;
2236 currentStyle.SetFlags(currentStyle.GetFlags() & ~wxTEXT_ATTR_BULLET_NUMBER);
2237 }
2238 }
2239 else
2240 currentStyle.SetBulletNumber(style.GetBulletNumber());
2241 }
2242
2243 if (style.HasBulletText() && !wxHasStyle(multipleStyleAttributes, wxTEXT_ATTR_BULLET_TEXT))
2244 {
2245 if (currentStyle.HasBulletText())
2246 {
2247 if (currentStyle.GetBulletText() != style.GetBulletText())
2248 {
2249 // Clash of style - mark as such
2250 multipleStyleAttributes |= wxTEXT_ATTR_BULLET_TEXT;
2251 currentStyle.SetFlags(currentStyle.GetFlags() & ~wxTEXT_ATTR_BULLET_TEXT);
2252 }
2253 }
2254 else
2255 {
2256 currentStyle.SetBulletText(style.GetBulletText());
2257 currentStyle.SetBulletFont(style.GetBulletFont());
2258 }
2259 }
2260
2261 if (style.HasBulletName() && !wxHasStyle(multipleStyleAttributes, wxTEXT_ATTR_BULLET_NAME))
2262 {
2263 if (currentStyle.HasBulletName())
2264 {
2265 if (currentStyle.GetBulletName() != style.GetBulletName())
2266 {
2267 // Clash of style - mark as such
2268 multipleStyleAttributes |= wxTEXT_ATTR_BULLET_NAME;
2269 currentStyle.SetFlags(currentStyle.GetFlags() & ~wxTEXT_ATTR_BULLET_NAME);
2270 }
2271 }
2272 else
2273 {
2274 currentStyle.SetBulletName(style.GetBulletName());
2275 }
2276 }
2277
2278 if (style.HasURL() && !wxHasStyle(multipleStyleAttributes, wxTEXT_ATTR_URL))
2279 {
2280 if (currentStyle.HasURL())
2281 {
2282 if (currentStyle.GetURL() != style.GetURL())
2283 {
2284 // Clash of style - mark as such
2285 multipleStyleAttributes |= wxTEXT_ATTR_URL;
2286 currentStyle.SetFlags(currentStyle.GetFlags() & ~wxTEXT_ATTR_URL);
2287 }
2288 }
2289 else
2290 {
2291 currentStyle.SetURL(style.GetURL());
2292 }
2293 }
2294
2295 if (style.HasTextEffects() && !wxHasStyle(multipleStyleAttributes, wxTEXT_ATTR_EFFECTS))
2296 {
2297 if (currentStyle.HasTextEffects())
2298 {
2299 // We need to find the bits in the new style that are different:
2300 // just look at those bits that are specified by the new style.
2301
2302 int currentRelevantTextEffects = currentStyle.GetTextEffects() & style.GetTextEffectFlags();
2303 int newRelevantTextEffects = style.GetTextEffects() & style.GetTextEffectFlags();
2304
2305 if (currentRelevantTextEffects != newRelevantTextEffects)
2306 {
2307 // Find the text effects that were different, using XOR
2308 int differentEffects = currentRelevantTextEffects ^ newRelevantTextEffects;
2309
2310 // Clash of style - mark as such
2311 multipleTextEffectAttributes |= differentEffects;
2312 currentStyle.SetTextEffectFlags(currentStyle.GetTextEffectFlags() & ~differentEffects);
2313 }
2314 }
2315 else
2316 {
2317 currentStyle.SetTextEffects(style.GetTextEffects());
2318 currentStyle.SetTextEffectFlags(style.GetTextEffectFlags());
2319 }
2320 }
2321
2322 if (style.HasOutlineLevel() && !wxHasStyle(multipleStyleAttributes, wxTEXT_ATTR_OUTLINE_LEVEL))
2323 {
2324 if (currentStyle.HasOutlineLevel())
2325 {
2326 if (currentStyle.GetOutlineLevel() != style.GetOutlineLevel())
2327 {
2328 // Clash of style - mark as such
2329 multipleStyleAttributes |= wxTEXT_ATTR_OUTLINE_LEVEL;
2330 currentStyle.SetFlags(currentStyle.GetFlags() & ~wxTEXT_ATTR_OUTLINE_LEVEL);
2331 }
2332 }
2333 else
2334 currentStyle.SetOutlineLevel(style.GetOutlineLevel());
2335 }
2336
2337 return true;
2338 }
2339
2340 /// Get the combined style for a range - if any attribute is different within the range,
2341 /// that attribute is not present within the flags.
2342 /// *** Note that this is not recursive, and so assumes that content inside a paragraph is not itself
2343 /// nested.
2344 bool wxRichTextParagraphLayoutBox::GetStyleForRange(const wxRichTextRange& range, wxTextAttr& style)
2345 {
2346 style = wxTextAttr();
2347
2348 // The attributes that aren't valid because of multiple styles within the range
2349 long multipleStyleAttributes = 0;
2350 int multipleTextEffectAttributes = 0;
2351
2352 wxRichTextObjectList::compatibility_iterator node = GetChildren().GetFirst();
2353 while (node)
2354 {
2355 wxRichTextParagraph* para = (wxRichTextParagraph*) node->GetData();
2356 if (!(para->GetRange().GetStart() > range.GetEnd() || para->GetRange().GetEnd() < range.GetStart()))
2357 {
2358 if (para->GetChildren().GetCount() == 0)
2359 {
2360 wxTextAttr paraStyle = para->GetCombinedAttributes();
2361
2362 CollectStyle(style, paraStyle, multipleStyleAttributes, multipleTextEffectAttributes);
2363 }
2364 else
2365 {
2366 wxRichTextRange paraRange(para->GetRange());
2367 paraRange.LimitTo(range);
2368
2369 // First collect paragraph attributes only
2370 wxTextAttr paraStyle = para->GetCombinedAttributes();
2371 paraStyle.SetFlags(paraStyle.GetFlags() & wxTEXT_ATTR_PARAGRAPH);
2372 CollectStyle(style, paraStyle, multipleStyleAttributes, multipleTextEffectAttributes);
2373
2374 wxRichTextObjectList::compatibility_iterator childNode = para->GetChildren().GetFirst();
2375
2376 while (childNode)
2377 {
2378 wxRichTextObject* child = childNode->GetData();
2379 if (!(child->GetRange().GetStart() > range.GetEnd() || child->GetRange().GetEnd() < range.GetStart()))
2380 {
2381 wxTextAttr childStyle = para->GetCombinedAttributes(child->GetAttributes());
2382
2383 // Now collect character attributes only
2384 childStyle.SetFlags(childStyle.GetFlags() & wxTEXT_ATTR_CHARACTER);
2385
2386 CollectStyle(style, childStyle, multipleStyleAttributes, multipleTextEffectAttributes);
2387 }
2388
2389 childNode = childNode->GetNext();
2390 }
2391 }
2392 }
2393 node = node->GetNext();
2394 }
2395 return true;
2396 }
2397
2398 /// Set default style
2399 bool wxRichTextParagraphLayoutBox::SetDefaultStyle(const wxTextAttr& style)
2400 {
2401 m_defaultAttributes = style;
2402 return true;
2403 }
2404
2405 /// Test if this whole range has character attributes of the specified kind. If any
2406 /// of the attributes are different within the range, the test fails. You
2407 /// can use this to implement, for example, bold button updating. style must have
2408 /// flags indicating which attributes are of interest.
2409 bool wxRichTextParagraphLayoutBox::HasCharacterAttributes(const wxRichTextRange& range, const wxTextAttr& style) const
2410 {
2411 int foundCount = 0;
2412 int matchingCount = 0;
2413
2414 wxRichTextObjectList::compatibility_iterator node = m_children.GetFirst();
2415 while (node)
2416 {
2417 wxRichTextParagraph* para = wxDynamicCast(node->GetData(), wxRichTextParagraph);
2418 wxASSERT (para != NULL);
2419
2420 if (para)
2421 {
2422 // Stop searching if we're beyond the range of interest
2423 if (para->GetRange().GetStart() > range.GetEnd())
2424 return foundCount == matchingCount;
2425
2426 if (!para->GetRange().IsOutside(range))
2427 {
2428 wxRichTextObjectList::compatibility_iterator node2 = para->GetChildren().GetFirst();
2429
2430 while (node2)
2431 {
2432 wxRichTextObject* child = node2->GetData();
2433 if (!child->GetRange().IsOutside(range) && child->IsKindOf(CLASSINFO(wxRichTextPlainText)))
2434 {
2435 foundCount ++;
2436 wxTextAttr textAttr = para->GetCombinedAttributes(child->GetAttributes());
2437
2438 if (wxTextAttrEqPartial(textAttr, style, style.GetFlags()))
2439 matchingCount ++;
2440 }
2441
2442 node2 = node2->GetNext();
2443 }
2444 }
2445 }
2446
2447 node = node->GetNext();
2448 }
2449
2450 return foundCount == matchingCount;
2451 }
2452
2453 /// Test if this whole range has paragraph attributes of the specified kind. If any
2454 /// of the attributes are different within the range, the test fails. You
2455 /// can use this to implement, for example, centering button updating. style must have
2456 /// flags indicating which attributes are of interest.
2457 bool wxRichTextParagraphLayoutBox::HasParagraphAttributes(const wxRichTextRange& range, const wxTextAttr& style) const
2458 {
2459 int foundCount = 0;
2460 int matchingCount = 0;
2461
2462 wxRichTextObjectList::compatibility_iterator node = m_children.GetFirst();
2463 while (node)
2464 {
2465 wxRichTextParagraph* para = wxDynamicCast(node->GetData(), wxRichTextParagraph);
2466 wxASSERT (para != NULL);
2467
2468 if (para)
2469 {
2470 // Stop searching if we're beyond the range of interest
2471 if (para->GetRange().GetStart() > range.GetEnd())
2472 return foundCount == matchingCount;
2473
2474 if (!para->GetRange().IsOutside(range))
2475 {
2476 wxTextAttr textAttr = GetAttributes();
2477 // Apply the paragraph style
2478 wxRichTextApplyStyle(textAttr, para->GetAttributes());
2479
2480 foundCount ++;
2481 if (wxTextAttrEqPartial(textAttr, style, style.GetFlags()))
2482 matchingCount ++;
2483 }
2484 }
2485
2486 node = node->GetNext();
2487 }
2488 return foundCount == matchingCount;
2489 }
2490
2491 void wxRichTextParagraphLayoutBox::Clear()
2492 {
2493 DeleteChildren();
2494 }
2495
2496 void wxRichTextParagraphLayoutBox::Reset()
2497 {
2498 Clear();
2499
2500 wxRichTextBuffer* buffer = wxDynamicCast(this, wxRichTextBuffer);
2501 if (buffer && GetRichTextCtrl())
2502 {
2503 wxRichTextEvent event(wxEVT_COMMAND_RICHTEXT_BUFFER_RESET, GetRichTextCtrl()->GetId());
2504 event.SetEventObject(GetRichTextCtrl());
2505
2506 buffer->SendEvent(event, true);
2507 }
2508
2509 AddParagraph(wxEmptyString);
2510
2511 Invalidate(wxRICHTEXT_ALL);
2512 }
2513
2514 /// Invalidate the buffer. With no argument, invalidates whole buffer.
2515 void wxRichTextParagraphLayoutBox::Invalidate(const wxRichTextRange& invalidRange)
2516 {
2517 SetDirty(true);
2518
2519 if (invalidRange == wxRICHTEXT_ALL)
2520 {
2521 m_invalidRange = wxRICHTEXT_ALL;
2522 return;
2523 }
2524
2525 // Already invalidating everything
2526 if (m_invalidRange == wxRICHTEXT_ALL)
2527 return;
2528
2529 if ((invalidRange.GetStart() < m_invalidRange.GetStart()) || m_invalidRange.GetStart() == -1)
2530 m_invalidRange.SetStart(invalidRange.GetStart());
2531 if (invalidRange.GetEnd() > m_invalidRange.GetEnd())
2532 m_invalidRange.SetEnd(invalidRange.GetEnd());
2533 }
2534
2535 /// Get invalid range, rounding to entire paragraphs if argument is true.
2536 wxRichTextRange wxRichTextParagraphLayoutBox::GetInvalidRange(bool wholeParagraphs) const
2537 {
2538 if (m_invalidRange == wxRICHTEXT_ALL || m_invalidRange == wxRICHTEXT_NONE)
2539 return m_invalidRange;
2540
2541 wxRichTextRange range = m_invalidRange;
2542
2543 if (wholeParagraphs)
2544 {
2545 wxRichTextParagraph* para1 = GetParagraphAtPosition(range.GetStart());
2546 wxRichTextParagraph* para2 = GetParagraphAtPosition(range.GetEnd());
2547 if (para1)
2548 range.SetStart(para1->GetRange().GetStart());
2549 if (para2)
2550 range.SetEnd(para2->GetRange().GetEnd());
2551 }
2552 return range;
2553 }
2554
2555 /// Apply the style sheet to the buffer, for example if the styles have changed.
2556 bool wxRichTextParagraphLayoutBox::ApplyStyleSheet(wxRichTextStyleSheet* styleSheet)
2557 {
2558 wxASSERT(styleSheet != NULL);
2559 if (!styleSheet)
2560 return false;
2561
2562 int foundCount = 0;
2563
2564 wxRichTextAttr attr(GetBasicStyle());
2565 if (GetBasicStyle().HasParagraphStyleName())
2566 {
2567 wxRichTextParagraphStyleDefinition* paraDef = styleSheet->FindParagraphStyle(GetBasicStyle().GetParagraphStyleName());
2568 if (paraDef)
2569 {
2570 attr.Apply(paraDef->GetStyleMergedWithBase(styleSheet));
2571 SetBasicStyle(attr);
2572 foundCount ++;
2573 }
2574 }
2575
2576 if (GetBasicStyle().HasCharacterStyleName())
2577 {
2578 wxRichTextCharacterStyleDefinition* charDef = styleSheet->FindCharacterStyle(GetBasicStyle().GetCharacterStyleName());
2579 if (charDef)
2580 {
2581 attr.Apply(charDef->GetStyleMergedWithBase(styleSheet));
2582 SetBasicStyle(attr);
2583 foundCount ++;
2584 }
2585 }
2586
2587 wxRichTextObjectList::compatibility_iterator node = m_children.GetFirst();
2588 while (node)
2589 {
2590 wxRichTextParagraph* para = wxDynamicCast(node->GetData(), wxRichTextParagraph);
2591 wxASSERT (para != NULL);
2592
2593 if (para)
2594 {
2595 // Combine paragraph and list styles. If there is a list style in the original attributes,
2596 // the current indentation overrides anything else and is used to find the item indentation.
2597 // Also, for applying paragraph styles, consider having 2 modes: (1) we merge with what we have,
2598 // thereby taking into account all user changes, (2) reset the style completely (except for indentation/list
2599 // exception as above).
2600 // Problem: when changing from one list style to another, there's a danger that the level info will get lost.
2601 // So when changing a list style interactively, could retrieve level based on current style, then
2602 // set appropriate indent and apply new style.
2603
2604 if (!para->GetAttributes().GetParagraphStyleName().IsEmpty() && !para->GetAttributes().GetListStyleName().IsEmpty())
2605 {
2606 int currentIndent = para->GetAttributes().GetLeftIndent();
2607
2608 wxRichTextParagraphStyleDefinition* paraDef = styleSheet->FindParagraphStyle(para->GetAttributes().GetParagraphStyleName());
2609 wxRichTextListStyleDefinition* listDef = styleSheet->FindListStyle(para->GetAttributes().GetListStyleName());
2610 if (paraDef && !listDef)
2611 {
2612 para->GetAttributes() = paraDef->GetStyleMergedWithBase(styleSheet);
2613 foundCount ++;
2614 }
2615 else if (listDef && !paraDef)
2616 {
2617 // Set overall style defined for the list style definition
2618 para->GetAttributes() = listDef->GetStyleMergedWithBase(styleSheet);
2619
2620 // Apply the style for this level
2621 wxRichTextApplyStyle(para->GetAttributes(), * listDef->GetLevelAttributes(listDef->FindLevelForIndent(currentIndent)));
2622 foundCount ++;
2623 }
2624 else if (listDef && paraDef)
2625 {
2626 // Combines overall list style, style for level, and paragraph style
2627 para->GetAttributes() = listDef->CombineWithParagraphStyle(currentIndent, paraDef->GetStyleMergedWithBase(styleSheet));
2628 foundCount ++;
2629 }
2630 }
2631 else if (para->GetAttributes().GetParagraphStyleName().IsEmpty() && !para->GetAttributes().GetListStyleName().IsEmpty())
2632 {
2633 int currentIndent = para->GetAttributes().GetLeftIndent();
2634
2635 wxRichTextListStyleDefinition* listDef = styleSheet->FindListStyle(para->GetAttributes().GetListStyleName());
2636
2637 // Overall list definition style
2638 para->GetAttributes() = listDef->GetStyleMergedWithBase(styleSheet);
2639
2640 // Style for this level
2641 wxRichTextApplyStyle(para->GetAttributes(), * listDef->GetLevelAttributes(listDef->FindLevelForIndent(currentIndent)));
2642
2643 foundCount ++;
2644 }
2645 else if (!para->GetAttributes().GetParagraphStyleName().IsEmpty() && para->GetAttributes().GetListStyleName().IsEmpty())
2646 {
2647 wxRichTextParagraphStyleDefinition* def = styleSheet->FindParagraphStyle(para->GetAttributes().GetParagraphStyleName());
2648 if (def)
2649 {
2650 para->GetAttributes() = def->GetStyleMergedWithBase(styleSheet);
2651 foundCount ++;
2652 }
2653 }
2654 }
2655
2656 node = node->GetNext();
2657 }
2658 return foundCount != 0;
2659 }
2660
2661 /// Set list style
2662 bool wxRichTextParagraphLayoutBox::SetListStyle(const wxRichTextRange& range, wxRichTextListStyleDefinition* def, int flags, int startFrom, int specifiedLevel)
2663 {
2664 wxRichTextStyleSheet* styleSheet = GetStyleSheet();
2665
2666 bool withUndo = ((flags & wxRICHTEXT_SETSTYLE_WITH_UNDO) != 0);
2667 // bool applyMinimal = ((flags & wxRICHTEXT_SETSTYLE_OPTIMIZE) != 0);
2668 bool specifyLevel = ((flags & wxRICHTEXT_SETSTYLE_SPECIFY_LEVEL) != 0);
2669 bool renumber = ((flags & wxRICHTEXT_SETSTYLE_RENUMBER) != 0);
2670
2671 // Current number, if numbering
2672 int n = startFrom;
2673
2674 wxASSERT (!specifyLevel || (specifyLevel && (specifiedLevel >= 0)));
2675
2676 // If we are associated with a control, make undoable; otherwise, apply immediately
2677 // to the data.
2678
2679 bool haveControl = (GetRichTextCtrl() != NULL);
2680
2681 wxRichTextAction* action = NULL;
2682
2683 if (haveControl && withUndo)
2684 {
2685 action = new wxRichTextAction(NULL, _("Change List Style"), wxRICHTEXT_CHANGE_STYLE, & GetRichTextCtrl()->GetBuffer(), GetRichTextCtrl());
2686 action->SetRange(range);
2687 action->SetPosition(GetRichTextCtrl()->GetCaretPosition());
2688 }
2689
2690 wxRichTextObjectList::compatibility_iterator node = m_children.GetFirst();
2691 while (node)
2692 {
2693 wxRichTextParagraph* para = wxDynamicCast(node->GetData(), wxRichTextParagraph);
2694 wxASSERT (para != NULL);
2695
2696 if (para && para->GetChildCount() > 0)
2697 {
2698 // Stop searching if we're beyond the range of interest
2699 if (para->GetRange().GetStart() > range.GetEnd())
2700 break;
2701
2702 if (!para->GetRange().IsOutside(range))
2703 {
2704 // We'll be using a copy of the paragraph to make style changes,
2705 // not updating the buffer directly.
2706 wxRichTextParagraph* newPara wxDUMMY_INITIALIZE(NULL);
2707
2708 if (haveControl && withUndo)
2709 {
2710 newPara = new wxRichTextParagraph(*para);
2711 action->GetNewParagraphs().AppendChild(newPara);
2712
2713 // Also store the old ones for Undo
2714 action->GetOldParagraphs().AppendChild(new wxRichTextParagraph(*para));
2715 }
2716 else
2717 newPara = para;
2718
2719 if (def)
2720 {
2721 int thisIndent = newPara->GetAttributes().GetLeftIndent();
2722 int thisLevel = specifyLevel ? specifiedLevel : def->FindLevelForIndent(thisIndent);
2723
2724 // How is numbering going to work?
2725 // If we are renumbering, or numbering for the first time, we need to keep
2726 // track of the number for each level. But we might be simply applying a different
2727 // list style.
2728 // In Word, applying a style to several paragraphs, even if at different levels,
2729 // reverts the level back to the same one. So we could do the same here.
2730 // Renumbering will need to be done when we promote/demote a paragraph.
2731
2732 // Apply the overall list style, and item style for this level
2733 wxTextAttr listStyle(def->GetCombinedStyleForLevel(thisLevel, styleSheet));
2734 wxRichTextApplyStyle(newPara->GetAttributes(), listStyle);
2735
2736 // Now we need to do numbering
2737 if (renumber)
2738 {
2739 newPara->GetAttributes().SetBulletNumber(n);
2740 }
2741
2742 n ++;
2743 }
2744 else if (!newPara->GetAttributes().GetListStyleName().IsEmpty())
2745 {
2746 // if def is NULL, remove list style, applying any associated paragraph style
2747 // to restore the attributes
2748
2749 newPara->GetAttributes().SetListStyleName(wxEmptyString);
2750 newPara->GetAttributes().SetLeftIndent(0, 0);
2751 newPara->GetAttributes().SetBulletText(wxEmptyString);
2752
2753 // Eliminate the main list-related attributes
2754 newPara->GetAttributes().SetFlags(newPara->GetAttributes().GetFlags() & ~wxTEXT_ATTR_LEFT_INDENT & ~wxTEXT_ATTR_BULLET_STYLE & ~wxTEXT_ATTR_BULLET_NUMBER & ~wxTEXT_ATTR_BULLET_TEXT & wxTEXT_ATTR_LIST_STYLE_NAME);
2755
2756 if (styleSheet && !newPara->GetAttributes().GetParagraphStyleName().IsEmpty())
2757 {
2758 wxRichTextParagraphStyleDefinition* def = styleSheet->FindParagraphStyle(newPara->GetAttributes().GetParagraphStyleName());
2759 if (def)
2760 {
2761 newPara->GetAttributes() = def->GetStyleMergedWithBase(styleSheet);
2762 }
2763 }
2764 }
2765 }
2766 }
2767
2768 node = node->GetNext();
2769 }
2770
2771 // Do action, or delay it until end of batch.
2772 if (haveControl && withUndo)
2773 GetRichTextCtrl()->GetBuffer().SubmitAction(action);
2774
2775 return true;
2776 }
2777
2778 bool wxRichTextParagraphLayoutBox::SetListStyle(const wxRichTextRange& range, const wxString& defName, int flags, int startFrom, int specifiedLevel)
2779 {
2780 if (GetStyleSheet())
2781 {
2782 wxRichTextListStyleDefinition* def = GetStyleSheet()->FindListStyle(defName);
2783 if (def)
2784 return SetListStyle(range, def, flags, startFrom, specifiedLevel);
2785 }
2786 return false;
2787 }
2788
2789 /// Clear list for given range
2790 bool wxRichTextParagraphLayoutBox::ClearListStyle(const wxRichTextRange& range, int flags)
2791 {
2792 return SetListStyle(range, NULL, flags);
2793 }
2794
2795 /// Number/renumber any list elements in the given range
2796 bool wxRichTextParagraphLayoutBox::NumberList(const wxRichTextRange& range, wxRichTextListStyleDefinition* def, int flags, int startFrom, int specifiedLevel)
2797 {
2798 return DoNumberList(range, range, 0, def, flags, startFrom, specifiedLevel);
2799 }
2800
2801 /// Number/renumber any list elements in the given range. Also do promotion or demotion of items, if specified
2802 bool wxRichTextParagraphLayoutBox::DoNumberList(const wxRichTextRange& range, const wxRichTextRange& promotionRange, int promoteBy,
2803 wxRichTextListStyleDefinition* def, int flags, int startFrom, int specifiedLevel)
2804 {
2805 wxRichTextStyleSheet* styleSheet = GetStyleSheet();
2806
2807 bool withUndo = ((flags & wxRICHTEXT_SETSTYLE_WITH_UNDO) != 0);
2808 // bool applyMinimal = ((flags & wxRICHTEXT_SETSTYLE_OPTIMIZE) != 0);
2809 #ifdef __WXDEBUG__
2810 bool specifyLevel = ((flags & wxRICHTEXT_SETSTYLE_SPECIFY_LEVEL) != 0);
2811 #endif
2812
2813 bool renumber = ((flags & wxRICHTEXT_SETSTYLE_RENUMBER) != 0);
2814
2815 // Max number of levels
2816 const int maxLevels = 10;
2817
2818 // The level we're looking at now
2819 int currentLevel = -1;
2820
2821 // The item number for each level
2822 int levels[maxLevels];
2823 int i;
2824
2825 // Reset all numbering
2826 for (i = 0; i < maxLevels; i++)
2827 {
2828 if (startFrom != -1)
2829 levels[i] = startFrom-1;
2830 else if (renumber) // start again
2831 levels[i] = 0;
2832 else
2833 levels[i] = -1; // start from the number we found, if any
2834 }
2835
2836 wxASSERT(!specifyLevel || (specifyLevel && (specifiedLevel >= 0)));
2837
2838 // If we are associated with a control, make undoable; otherwise, apply immediately
2839 // to the data.
2840
2841 bool haveControl = (GetRichTextCtrl() != NULL);
2842
2843 wxRichTextAction* action = NULL;
2844
2845 if (haveControl && withUndo)
2846 {
2847 action = new wxRichTextAction(NULL, _("Renumber List"), wxRICHTEXT_CHANGE_STYLE, & GetRichTextCtrl()->GetBuffer(), GetRichTextCtrl());
2848 action->SetRange(range);
2849 action->SetPosition(GetRichTextCtrl()->GetCaretPosition());
2850 }
2851
2852 wxRichTextObjectList::compatibility_iterator node = m_children.GetFirst();
2853 while (node)
2854 {
2855 wxRichTextParagraph* para = wxDynamicCast(node->GetData(), wxRichTextParagraph);
2856 wxASSERT (para != NULL);
2857
2858 if (para && para->GetChildCount() > 0)
2859 {
2860 // Stop searching if we're beyond the range of interest
2861 if (para->GetRange().GetStart() > range.GetEnd())
2862 break;
2863
2864 if (!para->GetRange().IsOutside(range))
2865 {
2866 // We'll be using a copy of the paragraph to make style changes,
2867 // not updating the buffer directly.
2868 wxRichTextParagraph* newPara wxDUMMY_INITIALIZE(NULL);
2869
2870 if (haveControl && withUndo)
2871 {
2872 newPara = new wxRichTextParagraph(*para);
2873 action->GetNewParagraphs().AppendChild(newPara);
2874
2875 // Also store the old ones for Undo
2876 action->GetOldParagraphs().AppendChild(new wxRichTextParagraph(*para));
2877 }
2878 else
2879 newPara = para;
2880
2881 wxRichTextListStyleDefinition* defToUse = def;
2882 if (!defToUse)
2883 {
2884 if (styleSheet && !newPara->GetAttributes().GetListStyleName().IsEmpty())
2885 defToUse = styleSheet->FindListStyle(newPara->GetAttributes().GetListStyleName());
2886 }
2887
2888 if (defToUse)
2889 {
2890 int thisIndent = newPara->GetAttributes().GetLeftIndent();
2891 int thisLevel = defToUse->FindLevelForIndent(thisIndent);
2892
2893 // If we've specified a level to apply to all, change the level.
2894 if (specifiedLevel != -1)
2895 thisLevel = specifiedLevel;
2896
2897 // Do promotion if specified
2898 if ((promoteBy != 0) && !para->GetRange().IsOutside(promotionRange))
2899 {
2900 thisLevel = thisLevel - promoteBy;
2901 if (thisLevel < 0)
2902 thisLevel = 0;
2903 if (thisLevel > 9)
2904 thisLevel = 9;
2905 }
2906
2907 // Apply the overall list style, and item style for this level
2908 wxTextAttr listStyle(defToUse->GetCombinedStyleForLevel(thisLevel, styleSheet));
2909 wxRichTextApplyStyle(newPara->GetAttributes(), listStyle);
2910
2911 // OK, we've (re)applied the style, now let's get the numbering right.
2912
2913 if (currentLevel == -1)
2914 currentLevel = thisLevel;
2915
2916 // Same level as before, do nothing except increment level's number afterwards
2917 if (currentLevel == thisLevel)
2918 {
2919 }
2920 // A deeper level: start renumbering all levels after current level
2921 else if (thisLevel > currentLevel)
2922 {
2923 for (i = currentLevel+1; i <= thisLevel; i++)
2924 {
2925 levels[i] = 0;
2926 }
2927 currentLevel = thisLevel;
2928 }
2929 else if (thisLevel < currentLevel)
2930 {
2931 currentLevel = thisLevel;
2932 }
2933
2934 // Use the current numbering if -1 and we have a bullet number already
2935 if (levels[currentLevel] == -1)
2936 {
2937 if (newPara->GetAttributes().HasBulletNumber())
2938 levels[currentLevel] = newPara->GetAttributes().GetBulletNumber();
2939 else
2940 levels[currentLevel] = 1;
2941 }
2942 else
2943 {
2944 levels[currentLevel] ++;
2945 }
2946
2947 newPara->GetAttributes().SetBulletNumber(levels[currentLevel]);
2948
2949 // Create the bullet text if an outline list
2950 if (listStyle.GetBulletStyle() & wxTEXT_ATTR_BULLET_STYLE_OUTLINE)
2951 {
2952 wxString text;
2953 for (i = 0; i <= currentLevel; i++)
2954 {
2955 if (!text.IsEmpty())
2956 text += wxT(".");
2957 text += wxString::Format(wxT("%d"), levels[i]);
2958 }
2959 newPara->GetAttributes().SetBulletText(text);
2960 }
2961 }
2962 }
2963 }
2964
2965 node = node->GetNext();
2966 }
2967
2968 // Do action, or delay it until end of batch.
2969 if (haveControl && withUndo)
2970 GetRichTextCtrl()->GetBuffer().SubmitAction(action);
2971
2972 return true;
2973 }
2974
2975 bool wxRichTextParagraphLayoutBox::NumberList(const wxRichTextRange& range, const wxString& defName, int flags, int startFrom, int specifiedLevel)
2976 {
2977 if (GetStyleSheet())
2978 {
2979 wxRichTextListStyleDefinition* def = NULL;
2980 if (!defName.IsEmpty())
2981 def = GetStyleSheet()->FindListStyle(defName);
2982 return NumberList(range, def, flags, startFrom, specifiedLevel);
2983 }
2984 return false;
2985 }
2986
2987 /// Promote the list items within the given range. promoteBy can be a positive or negative number, e.g. 1 or -1
2988 bool wxRichTextParagraphLayoutBox::PromoteList(int promoteBy, const wxRichTextRange& range, wxRichTextListStyleDefinition* def, int flags, int specifiedLevel)
2989 {
2990 // TODO
2991 // One strategy is to first work out the range within which renumbering must occur. Then could pass these two ranges
2992 // to NumberList with a flag indicating promotion is required within one of the ranges.
2993 // Find first and last paragraphs in range. Then for first, calculate new indentation and look back until we find
2994 // a paragraph that either has no list style, or has one that is different or whose indentation is less.
2995 // We start renumbering from the para after that different para we found. We specify that the numbering of that
2996 // list position will start from 1.
2997 // Similarly, we look after the last para in the promote range for an indentation that is less (or no list style).
2998 // We can end the renumbering at this point.
2999
3000 // For now, only renumber within the promotion range.
3001
3002 return DoNumberList(range, range, promoteBy, def, flags, 1, specifiedLevel);
3003 }
3004
3005 bool wxRichTextParagraphLayoutBox::PromoteList(int promoteBy, const wxRichTextRange& range, const wxString& defName, int flags, int specifiedLevel)
3006 {
3007 if (GetStyleSheet())
3008 {
3009 wxRichTextListStyleDefinition* def = NULL;
3010 if (!defName.IsEmpty())
3011 def = GetStyleSheet()->FindListStyle(defName);
3012 return PromoteList(promoteBy, range, def, flags, specifiedLevel);
3013 }
3014 return false;
3015 }
3016
3017 /// Fills in the attributes for numbering a paragraph after previousParagraph. It also finds the
3018 /// position of the paragraph that it had to start looking from.
3019 bool wxRichTextParagraphLayoutBox::FindNextParagraphNumber(wxRichTextParagraph* previousParagraph, wxTextAttr& attr) const
3020 {
3021 if (!previousParagraph->GetAttributes().HasFlag(wxTEXT_ATTR_BULLET_STYLE) || previousParagraph->GetAttributes().GetBulletStyle() == wxTEXT_ATTR_BULLET_STYLE_NONE)
3022 return false;
3023
3024 wxRichTextStyleSheet* styleSheet = GetStyleSheet();
3025 if (styleSheet && !previousParagraph->GetAttributes().GetListStyleName().IsEmpty())
3026 {
3027 wxRichTextListStyleDefinition* def = styleSheet->FindListStyle(previousParagraph->GetAttributes().GetListStyleName());
3028 if (def)
3029 {
3030 // int thisIndent = previousParagraph->GetAttributes().GetLeftIndent();
3031 // int thisLevel = def->FindLevelForIndent(thisIndent);
3032
3033 bool isOutline = (previousParagraph->GetAttributes().GetBulletStyle() & wxTEXT_ATTR_BULLET_STYLE_OUTLINE) != 0;
3034
3035 attr.SetFlags(previousParagraph->GetAttributes().GetFlags() & (wxTEXT_ATTR_BULLET_STYLE|wxTEXT_ATTR_BULLET_NUMBER|wxTEXT_ATTR_BULLET_TEXT|wxTEXT_ATTR_BULLET_NAME));
3036 if (previousParagraph->GetAttributes().HasBulletName())
3037 attr.SetBulletName(previousParagraph->GetAttributes().GetBulletName());
3038 attr.SetBulletStyle(previousParagraph->GetAttributes().GetBulletStyle());
3039 attr.SetListStyleName(previousParagraph->GetAttributes().GetListStyleName());
3040
3041 int nextNumber = previousParagraph->GetAttributes().GetBulletNumber() + 1;
3042 attr.SetBulletNumber(nextNumber);
3043
3044 if (isOutline)
3045 {
3046 wxString text = previousParagraph->GetAttributes().GetBulletText();
3047 if (!text.IsEmpty())
3048 {
3049 int pos = text.Find(wxT('.'), true);
3050 if (pos != wxNOT_FOUND)
3051 {
3052 text = text.Mid(0, text.Length() - pos - 1);
3053 }
3054 else
3055 text = wxEmptyString;
3056 if (!text.IsEmpty())
3057 text += wxT(".");
3058 text += wxString::Format(wxT("%d"), nextNumber);
3059 attr.SetBulletText(text);
3060 }
3061 }
3062
3063 return true;
3064 }
3065 else
3066 return false;
3067 }
3068 else
3069 return false;
3070 }
3071
3072 /*!
3073 * wxRichTextParagraph
3074 * This object represents a single paragraph (or in a straight text editor, a line).
3075 */
3076
3077 IMPLEMENT_DYNAMIC_CLASS(wxRichTextParagraph, wxRichTextBox)
3078
3079 wxArrayInt wxRichTextParagraph::sm_defaultTabs;
3080
3081 wxRichTextParagraph::wxRichTextParagraph(wxRichTextObject* parent, wxTextAttr* style):
3082 wxRichTextBox(parent)
3083 {
3084 if (style)
3085 SetAttributes(*style);
3086 }
3087
3088 wxRichTextParagraph::wxRichTextParagraph(const wxString& text, wxRichTextObject* parent, wxTextAttr* paraStyle, wxTextAttr* charStyle):
3089 wxRichTextBox(parent)
3090 {
3091 if (paraStyle)
3092 SetAttributes(*paraStyle);
3093
3094 AppendChild(new wxRichTextPlainText(text, this, charStyle));
3095 }
3096
3097 wxRichTextParagraph::~wxRichTextParagraph()
3098 {
3099 ClearLines();
3100 }
3101
3102 /// Draw the item
3103 bool wxRichTextParagraph::Draw(wxDC& dc, const wxRichTextRange& range, const wxRichTextRange& selectionRange, const wxRect& WXUNUSED(rect), int WXUNUSED(descent), int style)
3104 {
3105 wxTextAttr attr = GetCombinedAttributes();
3106
3107 // Draw the bullet, if any
3108 if (attr.GetBulletStyle() != wxTEXT_ATTR_BULLET_STYLE_NONE)
3109 {
3110 if (attr.GetLeftSubIndent() != 0)
3111 {
3112 int spaceBeforePara = ConvertTenthsMMToPixels(dc, attr.GetParagraphSpacingBefore());
3113 int leftIndent = ConvertTenthsMMToPixels(dc, attr.GetLeftIndent());
3114
3115 wxTextAttr bulletAttr(GetCombinedAttributes());
3116
3117 // Combine with the font of the first piece of content, if one is specified
3118 if (GetChildren().GetCount() > 0)
3119 {
3120 wxRichTextObject* firstObj = (wxRichTextObject*) GetChildren().GetFirst()->GetData();
3121 if (firstObj->GetAttributes().HasFont())
3122 {
3123 wxRichTextApplyStyle(bulletAttr, firstObj->GetAttributes());
3124 }
3125 }
3126
3127 // Get line height from first line, if any
3128 wxRichTextLine* line = m_cachedLines.GetFirst() ? (wxRichTextLine* ) m_cachedLines.GetFirst()->GetData() : (wxRichTextLine*) NULL;
3129
3130 wxPoint linePos;
3131 int lineHeight wxDUMMY_INITIALIZE(0);
3132 if (line)
3133 {
3134 lineHeight = line->GetSize().y;
3135 linePos = line->GetPosition() + GetPosition();
3136 }
3137 else
3138 {
3139 wxFont font;
3140 if (bulletAttr.HasFont() && GetBuffer())
3141 font = GetBuffer()->GetFontTable().FindFont(bulletAttr);
3142 else
3143 font = (*wxNORMAL_FONT);
3144
3145 wxCheckSetFont(dc, font);
3146
3147 lineHeight = dc.GetCharHeight();
3148 linePos = GetPosition();
3149 linePos.y += spaceBeforePara;
3150 }
3151
3152 wxRect bulletRect(GetPosition().x + leftIndent, linePos.y, linePos.x - (GetPosition().x + leftIndent), lineHeight);
3153
3154 if (attr.GetBulletStyle() & wxTEXT_ATTR_BULLET_STYLE_BITMAP)
3155 {
3156 if (wxRichTextBuffer::GetRenderer())
3157 wxRichTextBuffer::GetRenderer()->DrawBitmapBullet(this, dc, bulletAttr, bulletRect);
3158 }
3159 else if (attr.GetBulletStyle() & wxTEXT_ATTR_BULLET_STYLE_STANDARD)
3160 {
3161 if (wxRichTextBuffer::GetRenderer())
3162 wxRichTextBuffer::GetRenderer()->DrawStandardBullet(this, dc, bulletAttr, bulletRect);
3163 }
3164 else
3165 {
3166 wxString bulletText = GetBulletText();
3167
3168 if (!bulletText.empty() && wxRichTextBuffer::GetRenderer())
3169 wxRichTextBuffer::GetRenderer()->DrawTextBullet(this, dc, bulletAttr, bulletRect, bulletText);
3170 }
3171 }
3172 }
3173
3174 // Draw the range for each line, one object at a time.
3175
3176 wxRichTextLineList::compatibility_iterator node = m_cachedLines.GetFirst();
3177 while (node)
3178 {
3179 wxRichTextLine* line = node->GetData();
3180 wxRichTextRange lineRange = line->GetAbsoluteRange();
3181
3182 int maxDescent = line->GetDescent();
3183
3184 // Lines are specified relative to the paragraph
3185
3186 wxPoint linePosition = line->GetPosition() + GetPosition();
3187 wxPoint objectPosition = linePosition;
3188
3189 // Loop through objects until we get to the one within range
3190 wxRichTextObjectList::compatibility_iterator node2 = m_children.GetFirst();
3191 while (node2)
3192 {
3193 wxRichTextObject* child = node2->GetData();
3194
3195 if (!child->GetRange().IsOutside(lineRange) && !lineRange.IsOutside(range))
3196 {
3197 // Draw this part of the line at the correct position
3198 wxRichTextRange objectRange(child->GetRange());
3199 objectRange.LimitTo(lineRange);
3200
3201 wxSize objectSize;
3202 int descent = 0;
3203 child->GetRangeSize(objectRange, objectSize, descent, dc, wxRICHTEXT_UNFORMATTED, objectPosition);
3204
3205 // Use the child object's width, but the whole line's height
3206 wxRect childRect(objectPosition, wxSize(objectSize.x, line->GetSize().y));
3207 child->Draw(dc, objectRange, selectionRange, childRect, maxDescent, style);
3208
3209 objectPosition.x += objectSize.x;
3210 }
3211 else if (child->GetRange().GetStart() > lineRange.GetEnd())
3212 // Can break out of inner loop now since we've passed this line's range
3213 break;
3214
3215 node2 = node2->GetNext();
3216 }
3217
3218 node = node->GetNext();
3219 }
3220
3221 return true;
3222 }
3223
3224 /// Lay the item out
3225 bool wxRichTextParagraph::Layout(wxDC& dc, const wxRect& rect, int style)
3226 {
3227 wxTextAttr attr = GetCombinedAttributes();
3228
3229 // ClearLines();
3230
3231 // Increase the size of the paragraph due to spacing
3232 int spaceBeforePara = ConvertTenthsMMToPixels(dc, attr.GetParagraphSpacingBefore());
3233 int spaceAfterPara = ConvertTenthsMMToPixels(dc, attr.GetParagraphSpacingAfter());
3234 int leftIndent = ConvertTenthsMMToPixels(dc, attr.GetLeftIndent());
3235 int leftSubIndent = ConvertTenthsMMToPixels(dc, attr.GetLeftSubIndent());
3236 int rightIndent = ConvertTenthsMMToPixels(dc, attr.GetRightIndent());
3237
3238 int lineSpacing = 0;
3239
3240 // Let's assume line spacing of 10 is normal, 15 is 1.5, 20 is 2, etc.
3241 if (attr.GetLineSpacing() != 10 && GetBuffer())
3242 {
3243 wxFont font(GetBuffer()->GetFontTable().FindFont(attr));
3244 wxCheckSetFont(dc, font);
3245 lineSpacing = (ConvertTenthsMMToPixels(dc, dc.GetCharHeight()) * attr.GetLineSpacing())/10;
3246 }
3247
3248 // Available space for text on each line differs.
3249 int availableTextSpaceFirstLine = rect.GetWidth() - leftIndent - rightIndent;
3250
3251 // Bullets start the text at the same position as subsequent lines
3252 if (attr.GetBulletStyle() != wxTEXT_ATTR_BULLET_STYLE_NONE)
3253 availableTextSpaceFirstLine -= leftSubIndent;
3254
3255 int availableTextSpaceSubsequentLines = rect.GetWidth() - leftIndent - rightIndent - leftSubIndent;
3256
3257 // Start position for each line relative to the paragraph
3258 int startPositionFirstLine = leftIndent;
3259 int startPositionSubsequentLines = leftIndent + leftSubIndent;
3260
3261 // If we have a bullet in this paragraph, the start position for the first line's text
3262 // is actually leftIndent + leftSubIndent.
3263 if (attr.GetBulletStyle() != wxTEXT_ATTR_BULLET_STYLE_NONE)
3264 startPositionFirstLine = startPositionSubsequentLines;
3265
3266 long lastEndPos = GetRange().GetStart()-1;
3267 long lastCompletedEndPos = lastEndPos;
3268
3269 int currentWidth = 0;
3270 SetPosition(rect.GetPosition());
3271
3272 wxPoint currentPosition(0, spaceBeforePara); // We will calculate lines relative to paragraph
3273 int lineHeight = 0;
3274 int maxWidth = 0;
3275 int maxDescent = 0;
3276
3277 int lineCount = 0;
3278
3279 wxRichTextObjectList::compatibility_iterator node = m_children.GetFirst();
3280 while (node)
3281 {
3282 wxRichTextObject* child = node->GetData();
3283
3284 child->SetCachedSize(wxDefaultSize);
3285 child->Layout(dc, rect, style);
3286
3287 node = node->GetNext();
3288 }
3289
3290 #if wxRICHTEXT_USE_PARTIAL_TEXT_EXTENTS
3291 wxArrayInt partialExtents;
3292
3293 wxSize paraSize;
3294 int paraDescent;
3295
3296 // This calculates the partial text extents
3297 GetRangeSize(GetRange(), paraSize, paraDescent, dc, wxRICHTEXT_UNFORMATTED, wxPoint(0,0), & partialExtents);
3298 #endif
3299
3300 // Split up lines
3301
3302 // We may need to go back to a previous child, in which case create the new line,
3303 // find the child corresponding to the start position of the string, and
3304 // continue.
3305
3306 node = m_children.GetFirst();
3307 while (node)
3308 {
3309 wxRichTextObject* child = node->GetData();
3310
3311 // If this is e.g. a composite text box, it will need to be laid out itself.
3312 // But if just a text fragment or image, for example, this will
3313 // do nothing. NB: won't we need to set the position after layout?
3314 // since for example if position is dependent on vertical line size, we
3315 // can't tell the position until the size is determined. So possibly introduce
3316 // another layout phase.
3317
3318 // Available width depends on whether we're on the first or subsequent lines
3319 int availableSpaceForText = (lineCount == 0 ? availableTextSpaceFirstLine : availableTextSpaceSubsequentLines);
3320
3321 currentPosition.x = (lineCount == 0 ? startPositionFirstLine : startPositionSubsequentLines);
3322
3323 // We may only be looking at part of a child, if we searched back for wrapping
3324 // and found a suitable point some way into the child. So get the size for the fragment
3325 // if necessary.
3326
3327 long nextBreakPos = GetFirstLineBreakPosition(lastEndPos+1);
3328 long lastPosToUse = child->GetRange().GetEnd();
3329 bool lineBreakInThisObject = (nextBreakPos > -1 && nextBreakPos <= child->GetRange().GetEnd());
3330
3331 if (lineBreakInThisObject)
3332 lastPosToUse = nextBreakPos;
3333
3334 wxSize childSize;
3335 int childDescent = 0;
3336
3337 if ((nextBreakPos == -1) && (lastEndPos == child->GetRange().GetStart() - 1)) // i.e. we want to get the whole thing
3338 {
3339 childSize = child->GetCachedSize();
3340 childDescent = child->GetDescent();
3341 }
3342 else
3343 GetRangeSize(wxRichTextRange(lastEndPos+1, lastPosToUse), childSize, childDescent, dc, wxRICHTEXT_UNFORMATTED, rect.GetPosition());
3344
3345 // Cases:
3346 // 1) There was a line break BEFORE the natural break
3347 // 2) There was a line break AFTER the natural break
3348 // 3) The child still fits (carry on)
3349
3350 if ((lineBreakInThisObject && (childSize.x + currentWidth <= availableSpaceForText)) ||
3351 (childSize.x + currentWidth > availableSpaceForText))
3352 {
3353 long wrapPosition = 0;
3354
3355 // Find a place to wrap. This may walk back to previous children,
3356 // for example if a word spans several objects.
3357 if (!FindWrapPosition(wxRichTextRange(lastCompletedEndPos+1, child->GetRange().GetEnd()), dc, availableSpaceForText, wrapPosition, & partialExtents))
3358 {
3359 // If the function failed, just cut it off at the end of this child.
3360 wrapPosition = child->GetRange().GetEnd();
3361 }
3362
3363 // FindWrapPosition can still return a value that will put us in an endless wrapping loop
3364 if (wrapPosition <= lastCompletedEndPos)
3365 wrapPosition = wxMax(lastCompletedEndPos+1,child->GetRange().GetEnd());
3366
3367 // wxLogDebug(wxT("Split at %ld"), wrapPosition);
3368
3369 // Let's find the actual size of the current line now
3370 wxSize actualSize;
3371 wxRichTextRange actualRange(lastCompletedEndPos+1, wrapPosition);
3372 GetRangeSize(actualRange, actualSize, childDescent, dc, wxRICHTEXT_UNFORMATTED);
3373 currentWidth = actualSize.x;
3374 lineHeight = wxMax(lineHeight, actualSize.y);
3375 maxDescent = wxMax(childDescent, maxDescent);
3376
3377 // Add a new line
3378 wxRichTextLine* line = AllocateLine(lineCount);
3379
3380 // Set relative range so we won't have to change line ranges when paragraphs are moved
3381 line->SetRange(wxRichTextRange(actualRange.GetStart() - GetRange().GetStart(), actualRange.GetEnd() - GetRange().GetStart()));
3382 line->SetPosition(currentPosition);
3383 line->SetSize(wxSize(currentWidth, lineHeight));
3384 line->SetDescent(maxDescent);
3385
3386 // Now move down a line. TODO: add margins, spacing
3387 currentPosition.y += lineHeight;
3388 currentPosition.y += lineSpacing;
3389 currentWidth = 0;
3390 maxDescent = 0;
3391 maxWidth = wxMax(maxWidth, currentWidth);
3392
3393 lineCount ++;
3394
3395 // TODO: account for zero-length objects, such as fields
3396 wxASSERT(wrapPosition > lastCompletedEndPos);
3397
3398 lastEndPos = wrapPosition;
3399 lastCompletedEndPos = lastEndPos;
3400
3401 lineHeight = 0;
3402
3403 // May need to set the node back to a previous one, due to searching back in wrapping
3404 wxRichTextObject* childAfterWrapPosition = FindObjectAtPosition(wrapPosition+1);
3405 if (childAfterWrapPosition)
3406 node = m_children.Find(childAfterWrapPosition);
3407 else
3408 node = node->GetNext();
3409 }
3410 else
3411 {
3412 // We still fit, so don't add a line, and keep going
3413 currentWidth += childSize.x;
3414 lineHeight = wxMax(lineHeight, childSize.y);
3415 maxDescent = wxMax(childDescent, maxDescent);
3416
3417 maxWidth = wxMax(maxWidth, currentWidth);
3418 lastEndPos = child->GetRange().GetEnd();
3419
3420 node = node->GetNext();
3421 }
3422 }
3423
3424 // Add the last line - it's the current pos -> last para pos
3425 // Substract -1 because the last position is always the end-paragraph position.
3426 if (lastCompletedEndPos <= GetRange().GetEnd()-1)
3427 {
3428 currentPosition.x = (lineCount == 0 ? startPositionFirstLine : startPositionSubsequentLines);
3429
3430 wxRichTextLine* line = AllocateLine(lineCount);
3431
3432 wxRichTextRange actualRange(lastCompletedEndPos+1, GetRange().GetEnd()-1);
3433
3434 // Set relative range so we won't have to change line ranges when paragraphs are moved
3435 line->SetRange(wxRichTextRange(actualRange.GetStart() - GetRange().GetStart(), actualRange.GetEnd() - GetRange().GetStart()));
3436
3437 line->SetPosition(currentPosition);
3438
3439 if (lineHeight == 0 && GetBuffer())
3440 {
3441 wxFont font(GetBuffer()->GetFontTable().FindFont(attr));
3442 wxCheckSetFont(dc, font);
3443 lineHeight = dc.GetCharHeight();
3444 }
3445 if (maxDescent == 0)
3446 {
3447 int w, h;
3448 dc.GetTextExtent(wxT("X"), & w, &h, & maxDescent);
3449 }
3450
3451 line->SetSize(wxSize(currentWidth, lineHeight));
3452 line->SetDescent(maxDescent);
3453 currentPosition.y += lineHeight;
3454 currentPosition.y += lineSpacing;
3455 lineCount ++;
3456 }
3457
3458 // Remove remaining unused line objects, if any
3459 ClearUnusedLines(lineCount);
3460
3461 // Apply styles to wrapped lines
3462 ApplyParagraphStyle(attr, rect);
3463
3464 SetCachedSize(wxSize(maxWidth, currentPosition.y + spaceBeforePara + spaceAfterPara));
3465
3466 m_dirty = false;
3467
3468 return true;
3469 }
3470
3471 /// Apply paragraph styles, such as centering, to wrapped lines
3472 void wxRichTextParagraph::ApplyParagraphStyle(const wxTextAttr& attr, const wxRect& rect)
3473 {
3474 if (!attr.HasAlignment())
3475 return;
3476
3477 wxRichTextLineList::compatibility_iterator node = m_cachedLines.GetFirst();
3478 while (node)
3479 {
3480 wxRichTextLine* line = node->GetData();
3481
3482 wxPoint pos = line->GetPosition();
3483 wxSize size = line->GetSize();
3484
3485 // centering, right-justification
3486 if (attr.HasAlignment() && GetAttributes().GetAlignment() == wxTEXT_ALIGNMENT_CENTRE)
3487 {
3488 pos.x = (rect.GetWidth() - size.x)/2 + pos.x;
3489 line->SetPosition(pos);
3490 }
3491 else if (attr.HasAlignment() && GetAttributes().GetAlignment() == wxTEXT_ALIGNMENT_RIGHT)
3492 {
3493 pos.x = pos.x + rect.GetWidth() - size.x;
3494 line->SetPosition(pos);
3495 }
3496
3497 node = node->GetNext();
3498 }
3499 }
3500
3501 /// Insert text at the given position
3502 bool wxRichTextParagraph::InsertText(long pos, const wxString& text)
3503 {
3504 wxRichTextObject* childToUse = NULL;
3505 wxRichTextObjectList::compatibility_iterator nodeToUse = wxRichTextObjectList::compatibility_iterator();
3506
3507 wxRichTextObjectList::compatibility_iterator node = m_children.GetFirst();
3508 while (node)
3509 {
3510 wxRichTextObject* child = node->GetData();
3511 if (child->GetRange().Contains(pos) && child->GetRange().GetLength() > 0)
3512 {
3513 childToUse = child;
3514 nodeToUse = node;
3515 break;
3516 }
3517
3518 node = node->GetNext();
3519 }
3520
3521 if (childToUse)
3522 {
3523 wxRichTextPlainText* textObject = wxDynamicCast(childToUse, wxRichTextPlainText);
3524 if (textObject)
3525 {
3526 int posInString = pos - textObject->GetRange().GetStart();
3527
3528 wxString newText = textObject->GetText().Mid(0, posInString) +
3529 text + textObject->GetText().Mid(posInString);
3530 textObject->SetText(newText);
3531
3532 int textLength = text.length();
3533
3534 textObject->SetRange(wxRichTextRange(textObject->GetRange().GetStart(),
3535 textObject->GetRange().GetEnd() + textLength));
3536
3537 // Increment the end range of subsequent fragments in this paragraph.
3538 // We'll set the paragraph range itself at a higher level.
3539
3540 wxRichTextObjectList::compatibility_iterator node = nodeToUse->GetNext();
3541 while (node)
3542 {
3543 wxRichTextObject* child = node->GetData();
3544 child->SetRange(wxRichTextRange(textObject->GetRange().GetStart() + textLength,
3545 textObject->GetRange().GetEnd() + textLength));
3546
3547 node = node->GetNext();
3548 }
3549
3550 return true;
3551 }
3552 else
3553 {
3554 // TODO: if not a text object, insert at closest position, e.g. in front of it
3555 }
3556 }
3557 else
3558 {
3559 // Add at end.
3560 // Don't pass parent initially to suppress auto-setting of parent range.
3561 // We'll do that at a higher level.
3562 wxRichTextPlainText* textObject = new wxRichTextPlainText(text, this);
3563
3564 AppendChild(textObject);
3565 return true;
3566 }
3567
3568 return false;
3569 }
3570
3571 void wxRichTextParagraph::Copy(const wxRichTextParagraph& obj)
3572 {
3573 wxRichTextBox::Copy(obj);
3574 }
3575
3576 /// Clear the cached lines
3577 void wxRichTextParagraph::ClearLines()
3578 {
3579 WX_CLEAR_LIST(wxRichTextLineList, m_cachedLines);
3580 }
3581
3582 /// Get/set the object size for the given range. Returns false if the range
3583 /// is invalid for this object.
3584 bool wxRichTextParagraph::GetRangeSize(const wxRichTextRange& range, wxSize& size, int& descent, wxDC& dc, int flags, wxPoint position, wxArrayInt* partialExtents) const
3585 {
3586 if (!range.IsWithin(GetRange()))
3587 return false;
3588
3589 if (flags & wxRICHTEXT_UNFORMATTED)
3590 {
3591 // Just use unformatted data, assume no line breaks
3592 // TODO: take into account line breaks
3593
3594 wxSize sz;
3595
3596 wxArrayInt childExtents;
3597 wxArrayInt* p;
3598 if (partialExtents)
3599 p = & childExtents;
3600 else
3601 p = NULL;
3602
3603 wxRichTextObjectList::compatibility_iterator node = m_children.GetFirst();
3604 while (node)
3605 {
3606
3607 wxRichTextObject* child = node->GetData();
3608 if (!child->GetRange().IsOutside(range))
3609 {
3610 wxSize childSize;
3611
3612 wxRichTextRange rangeToUse = range;
3613 rangeToUse.LimitTo(child->GetRange());
3614 int childDescent = 0;
3615
3616 if (child->GetRangeSize(rangeToUse, childSize, childDescent, dc, flags, wxPoint(position.x + sz.x, position.y), p))
3617 {
3618 sz.y = wxMax(sz.y, childSize.y);
3619 sz.x += childSize.x;
3620 descent = wxMax(descent, childDescent);
3621
3622 if (partialExtents)
3623 {
3624 int lastSize;
3625 if (partialExtents->GetCount() > 0)
3626 lastSize = (*partialExtents)[partialExtents->GetCount()-1];
3627 else
3628 lastSize = 0;
3629
3630 size_t i;
3631 for (i = 0; i < childExtents.GetCount(); i++)
3632 {
3633 partialExtents->Add(childExtents[i] + lastSize);
3634 }
3635 }
3636 }
3637
3638 if (p)
3639 p->Clear();
3640 }
3641
3642 node = node->GetNext();
3643 }
3644 size = sz;
3645 }
3646 else
3647 {
3648 // Use formatted data, with line breaks
3649 wxSize sz;
3650
3651 // We're going to loop through each line, and then for each line,
3652 // call GetRangeSize for the fragment that comprises that line.
3653 // Only we have to do that multiple times within the line, because
3654 // the line may be broken into pieces. For now ignore line break commands
3655 // (so we can assume that getting the unformatted size for a fragment
3656 // within a line is the actual size)
3657
3658 wxRichTextLineList::compatibility_iterator node = m_cachedLines.GetFirst();
3659 while (node)
3660 {
3661 wxRichTextLine* line = node->GetData();
3662 wxRichTextRange lineRange = line->GetAbsoluteRange();
3663 if (!lineRange.IsOutside(range))
3664 {
3665 wxSize lineSize;
3666
3667 wxRichTextObjectList::compatibility_iterator node2 = m_children.GetFirst();
3668 while (node2)
3669 {
3670 wxRichTextObject* child = node2->GetData();
3671
3672 if (!child->GetRange().IsOutside(lineRange))
3673 {
3674 wxRichTextRange rangeToUse = lineRange;
3675 rangeToUse.LimitTo(child->GetRange());
3676
3677 wxSize childSize;
3678 int childDescent = 0;
3679 if (child->GetRangeSize(rangeToUse, childSize, childDescent, dc, flags, wxPoint(position.x + sz.x, position.y)))
3680 {
3681 lineSize.y = wxMax(lineSize.y, childSize.y);
3682 lineSize.x += childSize.x;
3683 }
3684 descent = wxMax(descent, childDescent);
3685 }
3686
3687 node2 = node2->GetNext();
3688 }
3689
3690 // Increase size by a line (TODO: paragraph spacing)
3691 sz.y += lineSize.y;
3692 sz.x = wxMax(sz.x, lineSize.x);
3693 }
3694 node = node->GetNext();
3695 }
3696 size = sz;
3697 }
3698 return true;
3699 }
3700
3701 /// Finds the absolute position and row height for the given character position
3702 bool wxRichTextParagraph::FindPosition(wxDC& dc, long index, wxPoint& pt, int* height, bool forceLineStart)
3703 {
3704 if (index == -1)
3705 {
3706 wxRichTextLine* line = ((wxRichTextParagraphLayoutBox*)GetParent())->GetLineAtPosition(0);
3707 if (line)
3708 *height = line->GetSize().y;
3709 else
3710 *height = dc.GetCharHeight();
3711
3712 // -1 means 'the start of the buffer'.
3713 pt = GetPosition();
3714 if (line)
3715 pt = pt + line->GetPosition();
3716
3717 return true;
3718 }
3719
3720 // The final position in a paragraph is taken to mean the position
3721 // at the start of the next paragraph.
3722 if (index == GetRange().GetEnd())
3723 {
3724 wxRichTextParagraphLayoutBox* parent = wxDynamicCast(GetParent(), wxRichTextParagraphLayoutBox);
3725 wxASSERT( parent != NULL );
3726
3727 // Find the height at the next paragraph, if any
3728 wxRichTextLine* line = parent->GetLineAtPosition(index + 1);
3729 if (line)
3730 {
3731 *height = line->GetSize().y;
3732 pt = line->GetAbsolutePosition();
3733 }
3734 else
3735 {
3736 *height = dc.GetCharHeight();
3737 int indent = ConvertTenthsMMToPixels(dc, m_attributes.GetLeftIndent());
3738 pt = wxPoint(indent, GetCachedSize().y);
3739 }
3740
3741 return true;
3742 }
3743
3744 if (index < GetRange().GetStart() || index > GetRange().GetEnd())
3745 return false;
3746
3747 wxRichTextLineList::compatibility_iterator node = m_cachedLines.GetFirst();
3748 while (node)
3749 {
3750 wxRichTextLine* line = node->GetData();
3751 wxRichTextRange lineRange = line->GetAbsoluteRange();
3752 if (index >= lineRange.GetStart() && index <= lineRange.GetEnd())
3753 {
3754 // If this is the last point in the line, and we're forcing the
3755 // returned value to be the start of the next line, do the required
3756 // thing.
3757 if (index == lineRange.GetEnd() && forceLineStart)
3758 {
3759 if (node->GetNext())
3760 {
3761 wxRichTextLine* nextLine = node->GetNext()->GetData();
3762 *height = nextLine->GetSize().y;
3763 pt = nextLine->GetAbsolutePosition();
3764 return true;
3765 }
3766 }
3767
3768 pt.y = line->GetPosition().y + GetPosition().y;
3769
3770 wxRichTextRange r(lineRange.GetStart(), index);
3771 wxSize rangeSize;
3772 int descent = 0;
3773
3774 // We find the size of the line up to this point,
3775 // then we can add this size to the line start position and
3776 // paragraph start position to find the actual position.
3777
3778 if (GetRangeSize(r, rangeSize, descent, dc, wxRICHTEXT_UNFORMATTED, line->GetPosition()+ GetPosition()))
3779 {
3780 pt.x = line->GetPosition().x + GetPosition().x + rangeSize.x;
3781 *height = line->GetSize().y;
3782
3783 return true;
3784 }
3785
3786 }
3787
3788 node = node->GetNext();
3789 }
3790
3791 return false;
3792 }
3793
3794 /// Hit-testing: returns a flag indicating hit test details, plus
3795 /// information about position
3796 int wxRichTextParagraph::HitTest(wxDC& dc, const wxPoint& pt, long& textPosition)
3797 {
3798 wxPoint paraPos = GetPosition();
3799
3800 wxRichTextLineList::compatibility_iterator node = m_cachedLines.GetFirst();
3801 while (node)
3802 {
3803 wxRichTextLine* line = node->GetData();
3804 wxPoint linePos = paraPos + line->GetPosition();
3805 wxSize lineSize = line->GetSize();
3806 wxRichTextRange lineRange = line->GetAbsoluteRange();
3807
3808 if (pt.y <= linePos.y + lineSize.y)
3809 {
3810 if (pt.x < linePos.x)
3811 {
3812 textPosition = lineRange.GetStart();
3813 return wxRICHTEXT_HITTEST_BEFORE|wxRICHTEXT_HITTEST_OUTSIDE;
3814 }
3815 else if (pt.x >= (linePos.x + lineSize.x))
3816 {
3817 textPosition = lineRange.GetEnd();
3818 return wxRICHTEXT_HITTEST_AFTER|wxRICHTEXT_HITTEST_OUTSIDE;
3819 }
3820 else
3821 {
3822 long i;
3823 int lastX = linePos.x;
3824 for (i = lineRange.GetStart(); i <= lineRange.GetEnd(); i++)
3825 {
3826 wxSize childSize;
3827 int descent = 0;
3828
3829 wxRichTextRange rangeToUse(lineRange.GetStart(), i);
3830
3831 GetRangeSize(rangeToUse, childSize, descent, dc, wxRICHTEXT_UNFORMATTED, linePos);
3832
3833 int nextX = childSize.x + linePos.x;
3834
3835 if (pt.x >= lastX && pt.x <= nextX)
3836 {
3837 textPosition = i;
3838
3839 // So now we know it's between i-1 and i.
3840 // Let's see if we can be more precise about
3841 // which side of the position it's on.
3842
3843 int midPoint = (nextX - lastX)/2 + lastX;
3844 if (pt.x >= midPoint)
3845 return wxRICHTEXT_HITTEST_AFTER;
3846 else
3847 return wxRICHTEXT_HITTEST_BEFORE;
3848 }
3849 else
3850 {
3851 lastX = nextX;
3852 }
3853 }
3854 }
3855 }
3856
3857 node = node->GetNext();
3858 }
3859
3860 return wxRICHTEXT_HITTEST_NONE;
3861 }
3862
3863 /// Split an object at this position if necessary, and return
3864 /// the previous object, or NULL if inserting at beginning.
3865 wxRichTextObject* wxRichTextParagraph::SplitAt(long pos, wxRichTextObject** previousObject)
3866 {
3867 wxRichTextObjectList::compatibility_iterator node = m_children.GetFirst();
3868 while (node)
3869 {
3870 wxRichTextObject* child = node->GetData();
3871
3872 if (pos == child->GetRange().GetStart())
3873 {
3874 if (previousObject)
3875 {
3876 if (node->GetPrevious())
3877 *previousObject = node->GetPrevious()->GetData();
3878 else
3879 *previousObject = NULL;
3880 }
3881
3882 return child;
3883 }
3884
3885 if (child->GetRange().Contains(pos))
3886 {
3887 // This should create a new object, transferring part of
3888 // the content to the old object and the rest to the new object.
3889 wxRichTextObject* newObject = child->DoSplit(pos);
3890
3891 // If we couldn't split this object, just insert in front of it.
3892 if (!newObject)
3893 {
3894 // Maybe this is an empty string, try the next one
3895 // return child;
3896 }
3897 else
3898 {
3899 // Insert the new object after 'child'
3900 if (node->GetNext())
3901 m_children.Insert(node->GetNext(), newObject);
3902 else
3903 m_children.Append(newObject);
3904 newObject->SetParent(this);
3905
3906 if (previousObject)
3907 *previousObject = child;
3908
3909 return newObject;
3910 }
3911 }
3912
3913 node = node->GetNext();
3914 }
3915 if (previousObject)
3916 *previousObject = NULL;
3917 return NULL;
3918 }
3919
3920 /// Move content to a list from obj on
3921 void wxRichTextParagraph::MoveToList(wxRichTextObject* obj, wxList& list)
3922 {
3923 wxRichTextObjectList::compatibility_iterator node = m_children.Find(obj);
3924 while (node)
3925 {
3926 wxRichTextObject* child = node->GetData();
3927 list.Append(child);
3928
3929 wxRichTextObjectList::compatibility_iterator oldNode = node;
3930
3931 node = node->GetNext();
3932
3933 m_children.DeleteNode(oldNode);
3934 }
3935 }
3936
3937 /// Add content back from list
3938 void wxRichTextParagraph::MoveFromList(wxList& list)
3939 {
3940 for (wxList::compatibility_iterator node = list.GetFirst(); node; node = node->GetNext())
3941 {
3942 AppendChild((wxRichTextObject*) node->GetData());
3943 }
3944 }
3945
3946 /// Calculate range
3947 void wxRichTextParagraph::CalculateRange(long start, long& end)
3948 {
3949 wxRichTextCompositeObject::CalculateRange(start, end);
3950
3951 // Add one for end of paragraph
3952 end ++;
3953
3954 m_range.SetRange(start, end);
3955 }
3956
3957 /// Find the object at the given position
3958 wxRichTextObject* wxRichTextParagraph::FindObjectAtPosition(long position)
3959 {
3960 wxRichTextObjectList::compatibility_iterator node = m_children.GetFirst();
3961 while (node)
3962 {
3963 wxRichTextObject* obj = node->GetData();
3964 if (obj->GetRange().Contains(position))
3965 return obj;
3966
3967 node = node->GetNext();
3968 }
3969 return NULL;
3970 }
3971
3972 /// Get the plain text searching from the start or end of the range.
3973 /// The resulting string may be shorter than the range given.
3974 bool wxRichTextParagraph::GetContiguousPlainText(wxString& text, const wxRichTextRange& range, bool fromStart)
3975 {
3976 text = wxEmptyString;
3977
3978 if (fromStart)
3979 {
3980 wxRichTextObjectList::compatibility_iterator node = m_children.GetFirst();
3981 while (node)
3982 {
3983 wxRichTextObject* obj = node->GetData();
3984 if (!obj->GetRange().IsOutside(range))
3985 {
3986 wxRichTextPlainText* textObj = wxDynamicCast(obj, wxRichTextPlainText);
3987 if (textObj)
3988 {
3989 text += textObj->GetTextForRange(range);
3990 }
3991 else
3992 return true;
3993 }
3994
3995 node = node->GetNext();
3996 }
3997 }
3998 else
3999 {
4000 wxRichTextObjectList::compatibility_iterator node = m_children.GetLast();
4001 while (node)
4002 {
4003 wxRichTextObject* obj = node->GetData();
4004 if (!obj->GetRange().IsOutside(range))
4005 {
4006 wxRichTextPlainText* textObj = wxDynamicCast(obj, wxRichTextPlainText);
4007 if (textObj)
4008 {
4009 text = textObj->GetTextForRange(range) + text;
4010 }
4011 else
4012 return true;
4013 }
4014
4015 node = node->GetPrevious();
4016 }
4017 }
4018
4019 return true;
4020 }
4021
4022 /// Find a suitable wrap position.
4023 bool wxRichTextParagraph::FindWrapPosition(const wxRichTextRange& range, wxDC& dc, int availableSpace, long& wrapPosition, wxArrayInt* partialExtents)
4024 {
4025 // Find the first position where the line exceeds the available space.
4026 wxSize sz;
4027 long breakPosition = range.GetEnd();
4028
4029 #if wxRICHTEXT_USE_PARTIAL_TEXT_EXTENTS
4030 if (partialExtents && partialExtents->GetCount() >= (size_t) (GetRange().GetLength()-1)) // the final position in a paragraph is the newline
4031 {
4032 int widthBefore;
4033
4034 if (range.GetStart() > GetRange().GetStart())
4035 widthBefore = (*partialExtents)[range.GetStart() - GetRange().GetStart() - 1];
4036 else
4037 widthBefore = 0;
4038
4039 size_t i;
4040 for (i = (size_t) range.GetStart(); i < (size_t) range.GetEnd(); i++)
4041 {
4042 int widthFromStartOfThisRange = (*partialExtents)[i - GetRange().GetStart()] - widthBefore;
4043
4044 if (widthFromStartOfThisRange >= availableSpace)
4045 {
4046 breakPosition = i-1;
4047 break;
4048 }
4049 }
4050 }
4051 else
4052 #endif
4053 {
4054 // Binary chop for speed
4055 long minPos = range.GetStart();
4056 long maxPos = range.GetEnd();
4057 while (true)
4058 {
4059 if (minPos == maxPos)
4060 {
4061 int descent = 0;
4062 GetRangeSize(wxRichTextRange(range.GetStart(), minPos), sz, descent, dc, wxRICHTEXT_UNFORMATTED);
4063
4064 if (sz.x > availableSpace)
4065 breakPosition = minPos - 1;
4066 break;
4067 }
4068 else if ((maxPos - minPos) == 1)
4069 {
4070 int descent = 0;
4071 GetRangeSize(wxRichTextRange(range.GetStart(), minPos), sz, descent, dc, wxRICHTEXT_UNFORMATTED);
4072
4073 if (sz.x > availableSpace)
4074 breakPosition = minPos - 1;
4075 else
4076 {
4077 GetRangeSize(wxRichTextRange(range.GetStart(), maxPos), sz, descent, dc, wxRICHTEXT_UNFORMATTED);
4078 if (sz.x > availableSpace)
4079 breakPosition = maxPos-1;
4080 }
4081 break;
4082 }
4083 else
4084 {
4085 long nextPos = minPos + ((maxPos - minPos) / 2);
4086
4087 int descent = 0;
4088 GetRangeSize(wxRichTextRange(range.GetStart(), nextPos), sz, descent, dc, wxRICHTEXT_UNFORMATTED);
4089
4090 if (sz.x > availableSpace)
4091 {
4092 maxPos = nextPos;
4093 }
4094 else
4095 {
4096 minPos = nextPos;
4097 }
4098 }
4099 }
4100 }
4101
4102 // Now we know the last position on the line.
4103 // Let's try to find a word break.
4104
4105 wxString plainText;
4106 if (GetContiguousPlainText(plainText, wxRichTextRange(range.GetStart(), breakPosition), false))
4107 {
4108 int newLinePos = plainText.Find(wxRichTextLineBreakChar);
4109 if (newLinePos != wxNOT_FOUND)
4110 {
4111 breakPosition = wxMax(0, range.GetStart() + newLinePos);
4112 }
4113 else
4114 {
4115 int spacePos = plainText.Find(wxT(' '), true);
4116 int tabPos = plainText.Find(wxT('\t'), true);
4117 int pos = wxMax(spacePos, tabPos);
4118 if (pos != wxNOT_FOUND)
4119 {
4120 int positionsFromEndOfString = plainText.length() - pos - 1;
4121 breakPosition = breakPosition - positionsFromEndOfString;
4122 }
4123 }
4124 }
4125
4126 wrapPosition = breakPosition;
4127
4128 return true;
4129 }
4130
4131 /// Get the bullet text for this paragraph.
4132 wxString wxRichTextParagraph::GetBulletText()
4133 {
4134 if (GetAttributes().GetBulletStyle() == wxTEXT_ATTR_BULLET_STYLE_NONE ||
4135 (GetAttributes().GetBulletStyle() & wxTEXT_ATTR_BULLET_STYLE_BITMAP))
4136 return wxEmptyString;
4137
4138 int number = GetAttributes().GetBulletNumber();
4139
4140 wxString text;
4141 if ((GetAttributes().GetBulletStyle() & wxTEXT_ATTR_BULLET_STYLE_ARABIC) || (GetAttributes().GetBulletStyle() & wxTEXT_ATTR_BULLET_STYLE_OUTLINE))
4142 {
4143 text.Printf(wxT("%d"), number);
4144 }
4145 else if (GetAttributes().GetBulletStyle() & wxTEXT_ATTR_BULLET_STYLE_LETTERS_UPPER)
4146 {
4147 // TODO: Unicode, and also check if number > 26
4148 text.Printf(wxT("%c"), (wxChar) (number+64));
4149 }
4150 else if (GetAttributes().GetBulletStyle() & wxTEXT_ATTR_BULLET_STYLE_LETTERS_LOWER)
4151 {
4152 // TODO: Unicode, and also check if number > 26
4153 text.Printf(wxT("%c"), (wxChar) (number+96));
4154 }
4155 else if (GetAttributes().GetBulletStyle() & wxTEXT_ATTR_BULLET_STYLE_ROMAN_UPPER)
4156 {
4157 text = wxRichTextDecimalToRoman(number);
4158 }
4159 else if (GetAttributes().GetBulletStyle() & wxTEXT_ATTR_BULLET_STYLE_ROMAN_LOWER)
4160 {
4161 text = wxRichTextDecimalToRoman(number);
4162 text.MakeLower();
4163 }
4164 else if (GetAttributes().GetBulletStyle() & wxTEXT_ATTR_BULLET_STYLE_SYMBOL)
4165 {
4166 text = GetAttributes().GetBulletText();
4167 }
4168
4169 if (GetAttributes().GetBulletStyle() & wxTEXT_ATTR_BULLET_STYLE_OUTLINE)
4170 {
4171 // The outline style relies on the text being computed statically,
4172 // since it depends on other levels points (e.g. 1.2.1.1). So normally the bullet text
4173 // should be stored in the attributes; if not, just use the number for this
4174 // level, as previously computed.
4175 if (!GetAttributes().GetBulletText().IsEmpty())
4176 text = GetAttributes().GetBulletText();
4177 }
4178
4179 if (GetAttributes().GetBulletStyle() & wxTEXT_ATTR_BULLET_STYLE_PARENTHESES)
4180 {
4181 text = wxT("(") + text + wxT(")");
4182 }
4183 else if (GetAttributes().GetBulletStyle() & wxTEXT_ATTR_BULLET_STYLE_RIGHT_PARENTHESIS)
4184 {
4185 text = text + wxT(")");
4186 }
4187
4188 if (GetAttributes().GetBulletStyle() & wxTEXT_ATTR_BULLET_STYLE_PERIOD)
4189 {
4190 text += wxT(".");
4191 }
4192
4193 return text;
4194 }
4195
4196 /// Allocate or reuse a line object
4197 wxRichTextLine* wxRichTextParagraph::AllocateLine(int pos)
4198 {
4199 if (pos < (int) m_cachedLines.GetCount())
4200 {
4201 wxRichTextLine* line = m_cachedLines.Item(pos)->GetData();
4202 line->Init(this);
4203 return line;
4204 }
4205 else
4206 {
4207 wxRichTextLine* line = new wxRichTextLine(this);
4208 m_cachedLines.Append(line);
4209 return line;
4210 }
4211 }
4212
4213 /// Clear remaining unused line objects, if any
4214 bool wxRichTextParagraph::ClearUnusedLines(int lineCount)
4215 {
4216 int cachedLineCount = m_cachedLines.GetCount();
4217 if ((int) cachedLineCount > lineCount)
4218 {
4219 for (int i = 0; i < (int) (cachedLineCount - lineCount); i ++)
4220 {
4221 wxRichTextLineList::compatibility_iterator node = m_cachedLines.GetLast();
4222 wxRichTextLine* line = node->GetData();
4223 m_cachedLines.Erase(node);
4224 delete line;
4225 }
4226 }
4227 return true;
4228 }
4229
4230 /// Get combined attributes of the base style, paragraph style and character style. We use this to dynamically
4231 /// retrieve the actual style.
4232 wxTextAttr wxRichTextParagraph::GetCombinedAttributes(const wxTextAttr& contentStyle) const
4233 {
4234 wxTextAttr attr;
4235 wxRichTextBuffer* buf = wxDynamicCast(GetParent(), wxRichTextBuffer);
4236 if (buf)
4237 {
4238 attr = buf->GetBasicStyle();
4239 wxRichTextApplyStyle(attr, GetAttributes());
4240 }
4241 else
4242 attr = GetAttributes();
4243
4244 wxRichTextApplyStyle(attr, contentStyle);
4245 return attr;
4246 }
4247
4248 /// Get combined attributes of the base style and paragraph style.
4249 wxTextAttr wxRichTextParagraph::GetCombinedAttributes() const
4250 {
4251 wxTextAttr attr;
4252 wxRichTextBuffer* buf = wxDynamicCast(GetParent(), wxRichTextBuffer);
4253 if (buf)
4254 {
4255 attr = buf->GetBasicStyle();
4256 wxRichTextApplyStyle(attr, GetAttributes());
4257 }
4258 else
4259 attr = GetAttributes();
4260
4261 return attr;
4262 }
4263
4264 /// Create default tabstop array
4265 void wxRichTextParagraph::InitDefaultTabs()
4266 {
4267 // create a default tab list at 10 mm each.
4268 for (int i = 0; i < 20; ++i)
4269 {
4270 sm_defaultTabs.Add(i*100);
4271 }
4272 }
4273
4274 /// Clear default tabstop array
4275 void wxRichTextParagraph::ClearDefaultTabs()
4276 {
4277 sm_defaultTabs.Clear();
4278 }
4279
4280 /// Get the first position from pos that has a line break character.
4281 long wxRichTextParagraph::GetFirstLineBreakPosition(long pos)
4282 {
4283 wxRichTextObjectList::compatibility_iterator node = m_children.GetFirst();
4284 while (node)
4285 {
4286 wxRichTextObject* obj = node->GetData();
4287 if (pos >= obj->GetRange().GetStart() && pos <= obj->GetRange().GetEnd())
4288 {
4289 wxRichTextPlainText* textObj = wxDynamicCast(obj, wxRichTextPlainText);
4290 if (textObj)
4291 {
4292 long breakPos = textObj->GetFirstLineBreakPosition(pos);
4293 if (breakPos > -1)
4294 return breakPos;
4295 }
4296 }
4297 node = node->GetNext();
4298 }
4299 return -1;
4300 }
4301
4302 /*!
4303 * wxRichTextLine
4304 * This object represents a line in a paragraph, and stores
4305 * offsets from the start of the paragraph representing the
4306 * start and end positions of the line.
4307 */
4308
4309 wxRichTextLine::wxRichTextLine(wxRichTextParagraph* parent)
4310 {
4311 Init(parent);
4312 }
4313
4314 /// Initialisation
4315 void wxRichTextLine::Init(wxRichTextParagraph* parent)
4316 {
4317 m_parent = parent;
4318 m_range.SetRange(-1, -1);
4319 m_pos = wxPoint(0, 0);
4320 m_size = wxSize(0, 0);
4321 m_descent = 0;
4322 }
4323
4324 /// Copy
4325 void wxRichTextLine::Copy(const wxRichTextLine& obj)
4326 {
4327 m_range = obj.m_range;
4328 }
4329
4330 /// Get the absolute object position
4331 wxPoint wxRichTextLine::GetAbsolutePosition() const
4332 {
4333 return m_parent->GetPosition() + m_pos;
4334 }
4335
4336 /// Get the absolute range
4337 wxRichTextRange wxRichTextLine::GetAbsoluteRange() const
4338 {
4339 wxRichTextRange range(m_range.GetStart() + m_parent->GetRange().GetStart(), 0);
4340 range.SetEnd(range.GetStart() + m_range.GetLength()-1);
4341 return range;
4342 }
4343
4344 /*!
4345 * wxRichTextPlainText
4346 * This object represents a single piece of text.
4347 */
4348
4349 IMPLEMENT_DYNAMIC_CLASS(wxRichTextPlainText, wxRichTextObject)
4350
4351 wxRichTextPlainText::wxRichTextPlainText(const wxString& text, wxRichTextObject* parent, wxTextAttr* style):
4352 wxRichTextObject(parent)
4353 {
4354 if (style)
4355 SetAttributes(*style);
4356
4357 m_text = text;
4358 }
4359
4360 #define USE_KERNING_FIX 1
4361
4362 // If insufficient tabs are defined, this is the tab width used
4363 #define WIDTH_FOR_DEFAULT_TABS 50
4364
4365 /// Draw the item
4366 bool wxRichTextPlainText::Draw(wxDC& dc, const wxRichTextRange& range, const wxRichTextRange& selectionRange, const wxRect& rect, int descent, int WXUNUSED(style))
4367 {
4368 wxRichTextParagraph* para = wxDynamicCast(GetParent(), wxRichTextParagraph);
4369 wxASSERT (para != NULL);
4370
4371 wxTextAttr textAttr(para ? para->GetCombinedAttributes(GetAttributes()) : GetAttributes());
4372
4373 int offset = GetRange().GetStart();
4374
4375 // Replace line break characters with spaces
4376 wxString str = m_text;
4377 wxString toRemove = wxRichTextLineBreakChar;
4378 str.Replace(toRemove, wxT(" "));
4379 if (textAttr.HasTextEffects() && (textAttr.GetTextEffects() & wxTEXT_ATTR_EFFECT_CAPITALS))
4380 str.MakeUpper();
4381
4382 long len = range.GetLength();
4383 wxString stringChunk = str.Mid(range.GetStart() - offset, (size_t) len);
4384
4385 // Test for the optimized situations where all is selected, or none
4386 // is selected.
4387
4388 wxFont textFont(GetBuffer()->GetFontTable().FindFont(textAttr));
4389 wxCheckSetFont(dc, textFont);
4390 int charHeight = dc.GetCharHeight();
4391
4392 int x, y;
4393 if ( textFont.Ok() )
4394 {
4395 if ( textAttr.HasTextEffects() && (textAttr.GetTextEffects() & wxTEXT_ATTR_EFFECT_SUPERSCRIPT) )
4396 {
4397 double size = static_cast<double>(textFont.GetPointSize()) / wxSCRIPT_MUL_FACTOR;
4398 textFont.SetPointSize( static_cast<int>(size) );
4399 x = rect.x;
4400 y = rect.y;
4401 wxCheckSetFont(dc, textFont);
4402 }
4403 else if ( textAttr.HasTextEffects() && (textAttr.GetTextEffects() & wxTEXT_ATTR_EFFECT_SUBSCRIPT) )
4404 {
4405 double size = static_cast<double>(textFont.GetPointSize()) / wxSCRIPT_MUL_FACTOR;
4406 textFont.SetPointSize( static_cast<int>(size) );
4407 x = rect.x;
4408 int sub_height = static_cast<int>( static_cast<double>(charHeight) / wxSCRIPT_MUL_FACTOR);
4409 y = rect.y + (rect.height - sub_height + (descent - m_descent));
4410 wxCheckSetFont(dc, textFont);
4411 }
4412 else
4413 {
4414 x = rect.x;
4415 y = rect.y + (rect.height - charHeight - (descent - m_descent));
4416 }
4417 }
4418 else
4419 {
4420 x = rect.x;
4421 y = rect.y + (rect.height - charHeight - (descent - m_descent));
4422 }
4423
4424 // (a) All selected.
4425 if (selectionRange.GetStart() <= range.GetStart() && selectionRange.GetEnd() >= range.GetEnd())
4426 {
4427 DrawTabbedString(dc, textAttr, rect, stringChunk, x, y, true);
4428 }
4429 // (b) None selected.
4430 else if (selectionRange.GetEnd() < range.GetStart() || selectionRange.GetStart() > range.GetEnd())
4431 {
4432 // Draw all unselected
4433 DrawTabbedString(dc, textAttr, rect, stringChunk, x, y, false);
4434 }
4435 else
4436 {
4437 // (c) Part selected, part not
4438 // Let's draw unselected chunk, selected chunk, then unselected chunk.
4439
4440 dc.SetBackgroundMode(wxBRUSHSTYLE_TRANSPARENT);
4441
4442 // 1. Initial unselected chunk, if any, up until start of selection.
4443 if (selectionRange.GetStart() > range.GetStart() && selectionRange.GetStart() <= range.GetEnd())
4444 {
4445 int r1 = range.GetStart();
4446 int s1 = selectionRange.GetStart()-1;
4447 int fragmentLen = s1 - r1 + 1;
4448 if (fragmentLen < 0)
4449 wxLogDebug(wxT("Mid(%d, %d"), (int)(r1 - offset), (int)fragmentLen);
4450 wxString stringFragment = str.Mid(r1 - offset, fragmentLen);
4451
4452 DrawTabbedString(dc, textAttr, rect, stringFragment, x, y, false);
4453
4454 #if USE_KERNING_FIX
4455 if (stringChunk.Find(wxT("\t")) == wxNOT_FOUND)
4456 {
4457 // Compensate for kerning difference
4458 wxString stringFragment2(str.Mid(r1 - offset, fragmentLen+1));
4459 wxString stringFragment3(str.Mid(r1 - offset + fragmentLen, 1));
4460
4461 wxCoord w1, h1, w2, h2, w3, h3;
4462 dc.GetTextExtent(stringFragment, & w1, & h1);
4463 dc.GetTextExtent(stringFragment2, & w2, & h2);
4464 dc.GetTextExtent(stringFragment3, & w3, & h3);
4465
4466 int kerningDiff = (w1 + w3) - w2;
4467 x = x - kerningDiff;
4468 }
4469 #endif
4470 }
4471
4472 // 2. Selected chunk, if any.
4473 if (selectionRange.GetEnd() >= range.GetStart())
4474 {
4475 int s1 = wxMax(selectionRange.GetStart(), range.GetStart());
4476 int s2 = wxMin(selectionRange.GetEnd(), range.GetEnd());
4477
4478 int fragmentLen = s2 - s1 + 1;
4479 if (fragmentLen < 0)
4480 wxLogDebug(wxT("Mid(%d, %d"), (int)(s1 - offset), (int)fragmentLen);
4481 wxString stringFragment = str.Mid(s1 - offset, fragmentLen);
4482
4483 DrawTabbedString(dc, textAttr, rect, stringFragment, x, y, true);
4484
4485 #if USE_KERNING_FIX
4486 if (stringChunk.Find(wxT("\t")) == wxNOT_FOUND)
4487 {
4488 // Compensate for kerning difference
4489 wxString stringFragment2(str.Mid(s1 - offset, fragmentLen+1));
4490 wxString stringFragment3(str.Mid(s1 - offset + fragmentLen, 1));
4491
4492 wxCoord w1, h1, w2, h2, w3, h3;
4493 dc.GetTextExtent(stringFragment, & w1, & h1);
4494 dc.GetTextExtent(stringFragment2, & w2, & h2);
4495 dc.GetTextExtent(stringFragment3, & w3, & h3);
4496
4497 int kerningDiff = (w1 + w3) - w2;
4498 x = x - kerningDiff;
4499 }
4500 #endif
4501 }
4502
4503 // 3. Remaining unselected chunk, if any
4504 if (selectionRange.GetEnd() < range.GetEnd())
4505 {
4506 int s2 = wxMin(selectionRange.GetEnd()+1, range.GetEnd());
4507 int r2 = range.GetEnd();
4508
4509 int fragmentLen = r2 - s2 + 1;
4510 if (fragmentLen < 0)
4511 wxLogDebug(wxT("Mid(%d, %d"), (int)(s2 - offset), (int)fragmentLen);
4512 wxString stringFragment = str.Mid(s2 - offset, fragmentLen);
4513
4514 DrawTabbedString(dc, textAttr, rect, stringFragment, x, y, false);
4515 }
4516 }
4517
4518 return true;
4519 }
4520
4521 bool wxRichTextPlainText::DrawTabbedString(wxDC& dc, const wxTextAttr& attr, const wxRect& rect,wxString& str, wxCoord& x, wxCoord& y, bool selected)
4522 {
4523 bool hasTabs = (str.Find(wxT('\t')) != wxNOT_FOUND);
4524
4525 wxArrayInt tabArray;
4526 int tabCount;
4527 if (hasTabs)
4528 {
4529 if (attr.GetTabs().IsEmpty())
4530 tabArray = wxRichTextParagraph::GetDefaultTabs();
4531 else
4532 tabArray = attr.GetTabs();
4533 tabCount = tabArray.GetCount();
4534
4535 for (int i = 0; i < tabCount; ++i)
4536 {
4537 int pos = tabArray[i];
4538 pos = ConvertTenthsMMToPixels(dc, pos);
4539 tabArray[i] = pos;
4540 }
4541 }
4542 else
4543 tabCount = 0;
4544
4545 int nextTabPos = -1;
4546 int tabPos = -1;
4547 wxCoord w, h;
4548
4549 if (selected)
4550 {
4551 wxColour highlightColour(wxSystemSettings::GetColour(wxSYS_COLOUR_HIGHLIGHT));
4552 wxColour highlightTextColour(wxSystemSettings::GetColour(wxSYS_COLOUR_HIGHLIGHTTEXT));
4553
4554 wxCheckSetBrush(dc, wxBrush(highlightColour));
4555 wxCheckSetPen(dc, wxPen(highlightColour));
4556 dc.SetTextForeground(highlightTextColour);
4557 dc.SetBackgroundMode(wxBRUSHSTYLE_TRANSPARENT);
4558 }
4559 else
4560 {
4561 dc.SetTextForeground(attr.GetTextColour());
4562
4563 if (attr.HasFlag(wxTEXT_ATTR_BACKGROUND_COLOUR) && attr.GetBackgroundColour().IsOk())
4564 {
4565 dc.SetBackgroundMode(wxBRUSHSTYLE_SOLID);
4566 dc.SetTextBackground(attr.GetBackgroundColour());
4567 }
4568 else
4569 dc.SetBackgroundMode(wxBRUSHSTYLE_TRANSPARENT);
4570 }
4571
4572 while (hasTabs)
4573 {
4574 // the string has a tab
4575 // break up the string at the Tab
4576 wxString stringChunk = str.BeforeFirst(wxT('\t'));
4577 str = str.AfterFirst(wxT('\t'));
4578 dc.GetTextExtent(stringChunk, & w, & h);
4579 tabPos = x + w;
4580 bool not_found = true;
4581 for (int i = 0; i < tabCount && not_found; ++i)
4582 {
4583 nextTabPos = tabArray.Item(i);
4584
4585 // Find the next tab position.
4586 // Even if we're at the end of the tab array, we must still draw the chunk.
4587
4588 if (nextTabPos > tabPos || (i == (tabCount - 1)))
4589 {
4590 if (nextTabPos <= tabPos)
4591 {
4592 int defaultTabWidth = ConvertTenthsMMToPixels(dc, WIDTH_FOR_DEFAULT_TABS);
4593 nextTabPos = tabPos + defaultTabWidth;
4594 }
4595
4596 not_found = false;
4597 if (selected)
4598 {
4599 w = nextTabPos - x;
4600 wxRect selRect(x, rect.y, w, rect.GetHeight());
4601 dc.DrawRectangle(selRect);
4602 }
4603 dc.DrawText(stringChunk, x, y);
4604
4605 if (attr.HasTextEffects() && (attr.GetTextEffects() & wxTEXT_ATTR_EFFECT_STRIKETHROUGH))
4606 {
4607 wxPen oldPen = dc.GetPen();
4608 wxCheckSetPen(dc, wxPen(attr.GetTextColour(), 1));
4609 dc.DrawLine(x, (int) (y+(h/2)+0.5), x+w, (int) (y+(h/2)+0.5));
4610 wxCheckSetPen(dc, oldPen);
4611 }
4612
4613 x = nextTabPos;
4614 }
4615 }
4616 hasTabs = (str.Find(wxT('\t')) != wxNOT_FOUND);
4617 }
4618
4619 if (!str.IsEmpty())
4620 {
4621 dc.GetTextExtent(str, & w, & h);
4622 if (selected)
4623 {
4624 wxRect selRect(x, rect.y, w, rect.GetHeight());
4625 dc.DrawRectangle(selRect);
4626 }
4627 dc.DrawText(str, x, y);
4628
4629 if (attr.HasTextEffects() && (attr.GetTextEffects() & wxTEXT_ATTR_EFFECT_STRIKETHROUGH))
4630 {
4631 wxPen oldPen = dc.GetPen();
4632 wxCheckSetPen(dc, wxPen(attr.GetTextColour(), 1));
4633 dc.DrawLine(x, (int) (y+(h/2)+0.5), x+w, (int) (y+(h/2)+0.5));
4634 wxCheckSetPen(dc, oldPen);
4635 }
4636
4637 x += w;
4638 }
4639 return true;
4640
4641 }
4642
4643 /// Lay the item out
4644 bool wxRichTextPlainText::Layout(wxDC& dc, const wxRect& WXUNUSED(rect), int WXUNUSED(style))
4645 {
4646 // Only lay out if we haven't already cached the size
4647 if (m_size.x == -1)
4648 GetRangeSize(GetRange(), m_size, m_descent, dc, 0, wxPoint(0, 0));
4649
4650 return true;
4651 }
4652
4653 /// Copy
4654 void wxRichTextPlainText::Copy(const wxRichTextPlainText& obj)
4655 {
4656 wxRichTextObject::Copy(obj);
4657
4658 m_text = obj.m_text;
4659 }
4660
4661 /// Get/set the object size for the given range. Returns false if the range
4662 /// is invalid for this object.
4663 bool wxRichTextPlainText::GetRangeSize(const wxRichTextRange& range, wxSize& size, int& descent, wxDC& dc, int WXUNUSED(flags), wxPoint position, wxArrayInt* partialExtents) const
4664 {
4665 if (!range.IsWithin(GetRange()))
4666 return false;
4667
4668 wxRichTextParagraph* para = wxDynamicCast(GetParent(), wxRichTextParagraph);
4669 wxASSERT (para != NULL);
4670
4671 wxTextAttr textAttr(para ? para->GetCombinedAttributes(GetAttributes()) : GetAttributes());
4672
4673 // Always assume unformatted text, since at this level we have no knowledge
4674 // of line breaks - and we don't need it, since we'll calculate size within
4675 // formatted text by doing it in chunks according to the line ranges
4676
4677 bool bScript(false);
4678 wxFont font(GetBuffer()->GetFontTable().FindFont(textAttr));
4679 if (font.Ok())
4680 {
4681 if ( textAttr.HasTextEffects() && ( (textAttr.GetTextEffects() & wxTEXT_ATTR_EFFECT_SUPERSCRIPT)
4682 || (textAttr.GetTextEffects() & wxTEXT_ATTR_EFFECT_SUBSCRIPT) ) )
4683 {
4684 wxFont textFont = font;
4685 double size = static_cast<double>(textFont.GetPointSize()) / wxSCRIPT_MUL_FACTOR;
4686 textFont.SetPointSize( static_cast<int>(size) );
4687 wxCheckSetFont(dc, textFont);
4688 bScript = true;
4689 }
4690 else
4691 {
4692 wxCheckSetFont(dc, font);
4693 }
4694 }
4695
4696 int startPos = range.GetStart() - GetRange().GetStart();
4697 long len = range.GetLength();
4698
4699 wxString str(m_text);
4700 wxString toReplace = wxRichTextLineBreakChar;
4701 str.Replace(toReplace, wxT(" "));
4702
4703 wxString stringChunk = str.Mid(startPos, (size_t) len);
4704
4705 if (textAttr.HasTextEffects() && (textAttr.GetTextEffects() & wxTEXT_ATTR_EFFECT_CAPITALS))
4706 stringChunk.MakeUpper();
4707
4708 wxCoord w, h;
4709 int width = 0;
4710 if (stringChunk.Find(wxT('\t')) != wxNOT_FOUND)
4711 {
4712 // the string has a tab
4713 wxArrayInt tabArray;
4714 if (textAttr.GetTabs().IsEmpty())
4715 tabArray = wxRichTextParagraph::GetDefaultTabs();
4716 else
4717 tabArray = textAttr.GetTabs();
4718
4719 int tabCount = tabArray.GetCount();
4720
4721 for (int i = 0; i < tabCount; ++i)
4722 {
4723 int pos = tabArray[i];
4724 pos = ((wxRichTextPlainText*) this)->ConvertTenthsMMToPixels(dc, pos);
4725 tabArray[i] = pos;
4726 }
4727
4728 int nextTabPos = -1;
4729
4730 while (stringChunk.Find(wxT('\t')) >= 0)
4731 {
4732 // the string has a tab
4733 // break up the string at the Tab
4734 wxString stringFragment = stringChunk.BeforeFirst(wxT('\t'));
4735 stringChunk = stringChunk.AfterFirst(wxT('\t'));
4736 int oldWidth = width;
4737 dc.GetTextExtent(stringFragment, & w, & h);
4738 width += w;
4739 int absoluteWidth = width + position.x;
4740
4741 if (partialExtents)
4742 {
4743 // Add these partial extents
4744 wxArrayInt p;
4745 dc.GetPartialTextExtents(stringFragment, p);
4746 size_t j;
4747 for (j = 0; j < p.GetCount(); j++)
4748 partialExtents->Add(oldWidth + p[j]);
4749 }
4750
4751 bool notFound = true;
4752 for (int i = 0; i < tabCount && notFound; ++i)
4753 {
4754 nextTabPos = tabArray.Item(i);
4755
4756 // Find the next tab position.
4757 // Even if we're at the end of the tab array, we must still process the chunk.
4758
4759 if (nextTabPos > absoluteWidth || (i == (tabCount - 1)))
4760 {
4761 if (nextTabPos <= absoluteWidth)
4762 {
4763 int defaultTabWidth = ((wxRichTextPlainText*) this)->ConvertTenthsMMToPixels(dc, WIDTH_FOR_DEFAULT_TABS);
4764 nextTabPos = absoluteWidth + defaultTabWidth;
4765 }
4766
4767 notFound = false;
4768 width = nextTabPos - position.x;
4769
4770 if (partialExtents)
4771 partialExtents->Add(width);
4772 }
4773 }
4774 }
4775 }
4776
4777 if (!stringChunk.IsEmpty())
4778 {
4779 dc.GetTextExtent(stringChunk, & w, & h, & descent);
4780 int oldWidth = width;
4781 width += w;
4782
4783 if (partialExtents)
4784 {
4785 // Add these partial extents
4786 wxArrayInt p;
4787 dc.GetPartialTextExtents(stringChunk, p);
4788 size_t j;
4789 for (j = 0; j < p.GetCount(); j++)
4790 partialExtents->Add(oldWidth + p[j]);
4791 }
4792 }
4793
4794 if ( bScript )
4795 dc.SetFont(font);
4796
4797 size = wxSize(width, dc.GetCharHeight());
4798
4799 return true;
4800 }
4801
4802 /// Do a split, returning an object containing the second part, and setting
4803 /// the first part in 'this'.
4804 wxRichTextObject* wxRichTextPlainText::DoSplit(long pos)
4805 {
4806 long index = pos - GetRange().GetStart();
4807
4808 if (index < 0 || index >= (int) m_text.length())
4809 return NULL;
4810
4811 wxString firstPart = m_text.Mid(0, index);
4812 wxString secondPart = m_text.Mid(index);
4813
4814 m_text = firstPart;
4815
4816 wxRichTextPlainText* newObject = new wxRichTextPlainText(secondPart);
4817 newObject->SetAttributes(GetAttributes());
4818
4819 newObject->SetRange(wxRichTextRange(pos, GetRange().GetEnd()));
4820 GetRange().SetEnd(pos-1);
4821
4822 return newObject;
4823 }
4824
4825 /// Calculate range
4826 void wxRichTextPlainText::CalculateRange(long start, long& end)
4827 {
4828 end = start + m_text.length() - 1;
4829 m_range.SetRange(start, end);
4830 }
4831
4832 /// Delete range
4833 bool wxRichTextPlainText::DeleteRange(const wxRichTextRange& range)
4834 {
4835 wxRichTextRange r = range;
4836
4837 r.LimitTo(GetRange());
4838
4839 if (r.GetStart() == GetRange().GetStart() && r.GetEnd() == GetRange().GetEnd())
4840 {
4841 m_text.Empty();
4842 return true;
4843 }
4844
4845 long startIndex = r.GetStart() - GetRange().GetStart();
4846 long len = r.GetLength();
4847
4848 m_text = m_text.Mid(0, startIndex) + m_text.Mid(startIndex+len);
4849 return true;
4850 }
4851
4852 /// Get text for the given range.
4853 wxString wxRichTextPlainText::GetTextForRange(const wxRichTextRange& range) const
4854 {
4855 wxRichTextRange r = range;
4856
4857 r.LimitTo(GetRange());
4858
4859 long startIndex = r.GetStart() - GetRange().GetStart();
4860 long len = r.GetLength();
4861
4862 return m_text.Mid(startIndex, len);
4863 }
4864
4865 /// Returns true if this object can merge itself with the given one.
4866 bool wxRichTextPlainText::CanMerge(wxRichTextObject* object) const
4867 {
4868 return object->GetClassInfo() == CLASSINFO(wxRichTextPlainText) &&
4869 (m_text.empty() || wxTextAttrEq(GetAttributes(), object->GetAttributes()));
4870 }
4871
4872 /// Returns true if this object merged itself with the given one.
4873 /// The calling code will then delete the given object.
4874 bool wxRichTextPlainText::Merge(wxRichTextObject* object)
4875 {
4876 wxRichTextPlainText* textObject = wxDynamicCast(object, wxRichTextPlainText);
4877 wxASSERT( textObject != NULL );
4878
4879 if (textObject)
4880 {
4881 m_text += textObject->GetText();
4882 wxRichTextApplyStyle(m_attributes, textObject->GetAttributes());
4883 return true;
4884 }
4885 else
4886 return false;
4887 }
4888
4889 /// Dump to output stream for debugging
4890 void wxRichTextPlainText::Dump(wxTextOutputStream& stream)
4891 {
4892 wxRichTextObject::Dump(stream);
4893 stream << m_text << wxT("\n");
4894 }
4895
4896 /// Get the first position from pos that has a line break character.
4897 long wxRichTextPlainText::GetFirstLineBreakPosition(long pos)
4898 {
4899 int i;
4900 int len = m_text.length();
4901 int startPos = pos - m_range.GetStart();
4902 for (i = startPos; i < len; i++)
4903 {
4904 wxChar ch = m_text[i];
4905 if (ch == wxRichTextLineBreakChar)
4906 {
4907 return i + m_range.GetStart();
4908 }
4909 }
4910 return -1;
4911 }
4912
4913 /*!
4914 * wxRichTextBuffer
4915 * This is a kind of box, used to represent the whole buffer
4916 */
4917
4918 IMPLEMENT_DYNAMIC_CLASS(wxRichTextBuffer, wxRichTextParagraphLayoutBox)
4919
4920 wxList wxRichTextBuffer::sm_handlers;
4921 wxRichTextRenderer* wxRichTextBuffer::sm_renderer = NULL;
4922 int wxRichTextBuffer::sm_bulletRightMargin = 20;
4923 float wxRichTextBuffer::sm_bulletProportion = (float) 0.3;
4924
4925 /// Initialisation
4926 void wxRichTextBuffer::Init()
4927 {
4928 m_commandProcessor = new wxCommandProcessor;
4929 m_styleSheet = NULL;
4930 m_modified = false;
4931 m_batchedCommandDepth = 0;
4932 m_batchedCommand = NULL;
4933 m_suppressUndo = 0;
4934 m_handlerFlags = 0;
4935 m_scale = 1.0;
4936 }
4937
4938 /// Initialisation
4939 wxRichTextBuffer::~wxRichTextBuffer()
4940 {
4941 delete m_commandProcessor;
4942 delete m_batchedCommand;
4943
4944 ClearStyleStack();
4945 ClearEventHandlers();
4946 }
4947
4948 void wxRichTextBuffer::ResetAndClearCommands()
4949 {
4950 Reset();
4951
4952 GetCommandProcessor()->ClearCommands();
4953
4954 Modify(false);
4955 Invalidate(wxRICHTEXT_ALL);
4956 }
4957
4958 void wxRichTextBuffer::Copy(const wxRichTextBuffer& obj)
4959 {
4960 wxRichTextParagraphLayoutBox::Copy(obj);
4961
4962 m_styleSheet = obj.m_styleSheet;
4963 m_modified = obj.m_modified;
4964 m_batchedCommandDepth = obj.m_batchedCommandDepth;
4965 m_batchedCommand = obj.m_batchedCommand;
4966 m_suppressUndo = obj.m_suppressUndo;
4967 }
4968
4969 /// Push style sheet to top of stack
4970 bool wxRichTextBuffer::PushStyleSheet(wxRichTextStyleSheet* styleSheet)
4971 {
4972 if (m_styleSheet)
4973 styleSheet->InsertSheet(m_styleSheet);
4974
4975 SetStyleSheet(styleSheet);
4976
4977 return true;
4978 }
4979
4980 /// Pop style sheet from top of stack
4981 wxRichTextStyleSheet* wxRichTextBuffer::PopStyleSheet()
4982 {
4983 if (m_styleSheet)
4984 {
4985 wxRichTextStyleSheet* oldSheet = m_styleSheet;
4986 m_styleSheet = oldSheet->GetNextSheet();
4987 oldSheet->Unlink();
4988
4989 return oldSheet;
4990 }
4991 else
4992 return NULL;
4993 }
4994
4995 /// Submit command to insert paragraphs
4996 bool wxRichTextBuffer::InsertParagraphsWithUndo(long pos, const wxRichTextParagraphLayoutBox& paragraphs, wxRichTextCtrl* ctrl, int flags)
4997 {
4998 wxRichTextAction* action = new wxRichTextAction(NULL, _("Insert Text"), wxRICHTEXT_INSERT, this, ctrl, false);
4999
5000 wxTextAttr attr(GetDefaultStyle());
5001
5002 wxTextAttr* p = NULL;
5003 wxTextAttr paraAttr;
5004 if (flags & wxRICHTEXT_INSERT_WITH_PREVIOUS_PARAGRAPH_STYLE)
5005 {
5006 paraAttr = GetStyleForNewParagraph(pos);
5007 if (!paraAttr.IsDefault())
5008 p = & paraAttr;
5009 }
5010 else
5011 p = & attr;
5012
5013 action->GetNewParagraphs() = paragraphs;
5014
5015 action->SetPosition(pos);
5016
5017 wxRichTextRange range = wxRichTextRange(pos, pos + paragraphs.GetRange().GetEnd() - 1);
5018 if (!paragraphs.GetPartialParagraph())
5019 range.SetEnd(range.GetEnd()+1);
5020
5021 // Set the range we'll need to delete in Undo
5022 action->SetRange(range);
5023
5024 SubmitAction(action);
5025
5026 return true;
5027 }
5028
5029 /// Submit command to insert the given text
5030 bool wxRichTextBuffer::InsertTextWithUndo(long pos, const wxString& text, wxRichTextCtrl* ctrl, int flags)
5031 {
5032 wxRichTextAction* action = new wxRichTextAction(NULL, _("Insert Text"), wxRICHTEXT_INSERT, this, ctrl, false);
5033
5034 wxTextAttr* p = NULL;
5035 wxTextAttr paraAttr;
5036 if (flags & wxRICHTEXT_INSERT_WITH_PREVIOUS_PARAGRAPH_STYLE)
5037 {
5038 // Get appropriate paragraph style
5039 paraAttr = GetStyleForNewParagraph(pos, false, false);
5040 if (!paraAttr.IsDefault())
5041 p = & paraAttr;
5042 }
5043
5044 action->GetNewParagraphs().AddParagraphs(text, p);
5045
5046 int length = action->GetNewParagraphs().GetRange().GetLength();
5047
5048 if (text.length() > 0 && text.Last() != wxT('\n'))
5049 {
5050 // Don't count the newline when undoing
5051 length --;
5052 action->GetNewParagraphs().SetPartialParagraph(true);
5053 }
5054 else if (text.length() > 0 && text.Last() == wxT('\n'))
5055 length --;
5056
5057 action->SetPosition(pos);
5058
5059 // Set the range we'll need to delete in Undo
5060 action->SetRange(wxRichTextRange(pos, pos + length - 1));
5061
5062 SubmitAction(action);
5063
5064 return true;
5065 }
5066
5067 /// Submit command to insert the given text
5068 bool wxRichTextBuffer::InsertNewlineWithUndo(long pos, wxRichTextCtrl* ctrl, int flags)
5069 {
5070 wxRichTextAction* action = new wxRichTextAction(NULL, _("Insert Text"), wxRICHTEXT_INSERT, this, ctrl, false);
5071
5072 wxTextAttr* p = NULL;
5073 wxTextAttr paraAttr;
5074 if (flags & wxRICHTEXT_INSERT_WITH_PREVIOUS_PARAGRAPH_STYLE)
5075 {
5076 paraAttr = GetStyleForNewParagraph(pos, false, true /* look for next paragraph style */);
5077 if (!paraAttr.IsDefault())
5078 p = & paraAttr;
5079 }
5080
5081 wxTextAttr attr(GetDefaultStyle());
5082
5083 wxRichTextParagraph* newPara = new wxRichTextParagraph(wxEmptyString, this, & attr);
5084 action->GetNewParagraphs().AppendChild(newPara);
5085 action->GetNewParagraphs().UpdateRanges();
5086 action->GetNewParagraphs().SetPartialParagraph(false);
5087 wxRichTextParagraph* para = GetParagraphAtPosition(pos, false);
5088 long pos1 = pos;
5089
5090 if (p)
5091 newPara->SetAttributes(*p);
5092
5093 if (flags & wxRICHTEXT_INSERT_INTERACTIVE)
5094 {
5095 if (para && para->GetRange().GetEnd() == pos)
5096 pos1 ++;
5097 if (newPara->GetAttributes().HasBulletNumber())
5098 newPara->GetAttributes().SetBulletNumber(newPara->GetAttributes().GetBulletNumber()+1);
5099 }
5100
5101 action->SetPosition(pos);
5102
5103 // Use the default character style
5104 // Use the default character style
5105 if (!GetDefaultStyle().IsDefault() && newPara->GetChildren().GetFirst())
5106 {
5107 // Check whether the default style merely reflects the paragraph/basic style,
5108 // in which case don't apply it.
5109 wxTextAttrEx defaultStyle(GetDefaultStyle());
5110 wxTextAttrEx toApply;
5111 if (para)
5112 {
5113 wxRichTextAttr combinedAttr = para->GetCombinedAttributes();
5114 wxTextAttrEx newAttr;
5115 // This filters out attributes that are accounted for by the current
5116 // paragraph/basic style
5117 wxRichTextApplyStyle(toApply, defaultStyle, & combinedAttr);
5118 }
5119 else
5120 toApply = defaultStyle;
5121
5122 if (!toApply.IsDefault())
5123 newPara->GetChildren().GetFirst()->GetData()->SetAttributes(toApply);
5124 }
5125
5126 // Set the range we'll need to delete in Undo
5127 action->SetRange(wxRichTextRange(pos1, pos1));
5128
5129 SubmitAction(action);
5130
5131 return true;
5132 }
5133
5134 /// Submit command to insert the given image
5135 bool wxRichTextBuffer::InsertImageWithUndo(long pos, const wxRichTextImageBlock& imageBlock, wxRichTextCtrl* ctrl, int flags)
5136 {
5137 wxRichTextAction* action = new wxRichTextAction(NULL, _("Insert Image"), wxRICHTEXT_INSERT, this, ctrl, false);
5138
5139 wxTextAttr* p = NULL;
5140 wxTextAttr paraAttr;
5141 if (flags & wxRICHTEXT_INSERT_WITH_PREVIOUS_PARAGRAPH_STYLE)
5142 {
5143 paraAttr = GetStyleForNewParagraph(pos);
5144 if (!paraAttr.IsDefault())
5145 p = & paraAttr;
5146 }
5147
5148 wxTextAttr attr(GetDefaultStyle());
5149
5150 wxRichTextParagraph* newPara = new wxRichTextParagraph(this, & attr);
5151 if (p)
5152 newPara->SetAttributes(*p);
5153
5154 wxRichTextImage* imageObject = new wxRichTextImage(imageBlock, newPara);
5155 newPara->AppendChild(imageObject);
5156 action->GetNewParagraphs().AppendChild(newPara);
5157 action->GetNewParagraphs().UpdateRanges();
5158
5159 action->GetNewParagraphs().SetPartialParagraph(true);
5160
5161 action->SetPosition(pos);
5162
5163 // Set the range we'll need to delete in Undo
5164 action->SetRange(wxRichTextRange(pos, pos));
5165
5166 SubmitAction(action);
5167
5168 return true;
5169 }
5170
5171 /// Get the style that is appropriate for a new paragraph at this position.
5172 /// If the previous paragraph has a paragraph style name, look up the next-paragraph
5173 /// style.
5174 wxTextAttr wxRichTextBuffer::GetStyleForNewParagraph(long pos, bool caretPosition, bool lookUpNewParaStyle) const
5175 {
5176 wxRichTextParagraph* para = GetParagraphAtPosition(pos, caretPosition);
5177 if (para)
5178 {
5179 wxTextAttr attr;
5180 bool foundAttributes = false;
5181
5182 // Look for a matching paragraph style
5183 if (lookUpNewParaStyle && !para->GetAttributes().GetParagraphStyleName().IsEmpty() && GetStyleSheet())
5184 {
5185 wxRichTextParagraphStyleDefinition* paraDef = GetStyleSheet()->FindParagraphStyle(para->GetAttributes().GetParagraphStyleName());
5186 if (paraDef)
5187 {
5188 // If we're not at the end of the paragraph, then we apply THIS style, and not the designated next style.
5189 if (para->GetRange().GetEnd() == pos && !paraDef->GetNextStyle().IsEmpty())
5190 {
5191 wxRichTextParagraphStyleDefinition* nextParaDef = GetStyleSheet()->FindParagraphStyle(paraDef->GetNextStyle());
5192 if (nextParaDef)
5193 {
5194 foundAttributes = true;
5195 attr = nextParaDef->GetStyleMergedWithBase(GetStyleSheet());
5196 }
5197 }
5198
5199 // If we didn't find the 'next style', use this style instead.
5200 if (!foundAttributes)
5201 {
5202 foundAttributes = true;
5203 attr = paraDef->GetStyleMergedWithBase(GetStyleSheet());
5204 }
5205 }
5206 }
5207 if (!foundAttributes)
5208 {
5209 attr = para->GetAttributes();
5210 int flags = attr.GetFlags();
5211
5212 // Eliminate character styles
5213 flags &= ( (~ wxTEXT_ATTR_FONT) |
5214 (~ wxTEXT_ATTR_TEXT_COLOUR) |
5215 (~ wxTEXT_ATTR_BACKGROUND_COLOUR) );
5216 attr.SetFlags(flags);
5217 }
5218
5219 // Now see if we need to number the paragraph.
5220 if (attr.HasBulletStyle())
5221 {
5222 wxTextAttr numberingAttr;
5223 if (FindNextParagraphNumber(para, numberingAttr))
5224 wxRichTextApplyStyle(attr, (const wxTextAttr&) numberingAttr);
5225 }
5226
5227 return attr;
5228 }
5229 else
5230 return wxTextAttr();
5231 }
5232
5233 /// Submit command to delete this range
5234 bool wxRichTextBuffer::DeleteRangeWithUndo(const wxRichTextRange& range, wxRichTextCtrl* ctrl)
5235 {
5236 wxRichTextAction* action = new wxRichTextAction(NULL, _("Delete"), wxRICHTEXT_DELETE, this, ctrl);
5237
5238 action->SetPosition(ctrl->GetCaretPosition());
5239
5240 // Set the range to delete
5241 action->SetRange(range);
5242
5243 // Copy the fragment that we'll need to restore in Undo
5244 CopyFragment(range, action->GetOldParagraphs());
5245
5246 // See if we're deleting a paragraph marker, in which case we need to
5247 // make a note not to copy the attributes from the 2nd paragraph to the 1st.
5248 if (range.GetStart() == range.GetEnd())
5249 {
5250 wxRichTextParagraph* para = GetParagraphAtPosition(range.GetStart());
5251 if (para && para->GetRange().GetEnd() == range.GetEnd())
5252 {
5253 wxRichTextParagraph* nextPara = GetParagraphAtPosition(range.GetStart()+1);
5254 if (nextPara && nextPara != para)
5255 {
5256 action->GetOldParagraphs().GetChildren().GetFirst()->GetData()->SetAttributes(nextPara->GetAttributes());
5257 action->GetOldParagraphs().GetAttributes().SetFlags(action->GetOldParagraphs().GetAttributes().GetFlags() | wxTEXT_ATTR_KEEP_FIRST_PARA_STYLE);
5258 }
5259 }
5260 }
5261
5262 SubmitAction(action);
5263
5264 return true;
5265 }
5266
5267 /// Collapse undo/redo commands
5268 bool wxRichTextBuffer::BeginBatchUndo(const wxString& cmdName)
5269 {
5270 if (m_batchedCommandDepth == 0)
5271 {
5272 wxASSERT(m_batchedCommand == NULL);
5273 if (m_batchedCommand)
5274 {
5275 GetCommandProcessor()->Store(m_batchedCommand);
5276 }
5277 m_batchedCommand = new wxRichTextCommand(cmdName);
5278 }
5279
5280 m_batchedCommandDepth ++;
5281
5282 return true;
5283 }
5284
5285 /// Collapse undo/redo commands
5286 bool wxRichTextBuffer::EndBatchUndo()
5287 {
5288 m_batchedCommandDepth --;
5289
5290 wxASSERT(m_batchedCommandDepth >= 0);
5291 wxASSERT(m_batchedCommand != NULL);
5292
5293 if (m_batchedCommandDepth == 0)
5294 {
5295 GetCommandProcessor()->Store(m_batchedCommand);
5296 m_batchedCommand = NULL;
5297 }
5298
5299 return true;
5300 }
5301
5302 /// Submit immediately, or delay according to whether collapsing is on
5303 bool wxRichTextBuffer::SubmitAction(wxRichTextAction* action)
5304 {
5305 if (BatchingUndo() && m_batchedCommand && !SuppressingUndo())
5306 {
5307 wxRichTextCommand* cmd = new wxRichTextCommand(action->GetName());
5308 cmd->AddAction(action);
5309 cmd->Do();
5310 cmd->GetActions().Clear();
5311 delete cmd;
5312
5313 m_batchedCommand->AddAction(action);
5314 }
5315 else
5316 {
5317 wxRichTextCommand* cmd = new wxRichTextCommand(action->GetName());
5318 cmd->AddAction(action);
5319
5320 // Only store it if we're not suppressing undo.
5321 return GetCommandProcessor()->Submit(cmd, !SuppressingUndo());
5322 }
5323
5324 return true;
5325 }
5326
5327 /// Begin suppressing undo/redo commands.
5328 bool wxRichTextBuffer::BeginSuppressUndo()
5329 {
5330 m_suppressUndo ++;
5331
5332 return true;
5333 }
5334
5335 /// End suppressing undo/redo commands.
5336 bool wxRichTextBuffer::EndSuppressUndo()
5337 {
5338 m_suppressUndo --;
5339
5340 return true;
5341 }
5342
5343 /// Begin using a style
5344 bool wxRichTextBuffer::BeginStyle(const wxTextAttr& style)
5345 {
5346 wxTextAttr newStyle(GetDefaultStyle());
5347
5348 // Save the old default style
5349 m_attributeStack.Append((wxObject*) new wxTextAttr(GetDefaultStyle()));
5350
5351 wxRichTextApplyStyle(newStyle, style);
5352 newStyle.SetFlags(style.GetFlags()|newStyle.GetFlags());
5353
5354 SetDefaultStyle(newStyle);
5355
5356 // wxLogDebug("Default style size = %d", GetDefaultStyle().GetFont().GetPointSize());
5357
5358 return true;
5359 }
5360
5361 /// End the style
5362 bool wxRichTextBuffer::EndStyle()
5363 {
5364 if (!m_attributeStack.GetFirst())
5365 {
5366 wxLogDebug(_("Too many EndStyle calls!"));
5367 return false;
5368 }
5369
5370 wxList::compatibility_iterator node = m_attributeStack.GetLast();
5371 wxTextAttr* attr = (wxTextAttr*)node->GetData();
5372 m_attributeStack.Erase(node);
5373
5374 SetDefaultStyle(*attr);
5375
5376 delete attr;
5377 return true;
5378 }
5379
5380 /// End all styles
5381 bool wxRichTextBuffer::EndAllStyles()
5382 {
5383 while (m_attributeStack.GetCount() != 0)
5384 EndStyle();
5385 return true;
5386 }
5387
5388 /// Clear the style stack
5389 void wxRichTextBuffer::ClearStyleStack()
5390 {
5391 for (wxList::compatibility_iterator node = m_attributeStack.GetFirst(); node; node = node->GetNext())
5392 delete (wxTextAttr*) node->GetData();
5393 m_attributeStack.Clear();
5394 }
5395
5396 /// Begin using bold
5397 bool wxRichTextBuffer::BeginBold()
5398 {
5399 wxTextAttr attr;
5400 attr.SetFontWeight(wxBOLD);
5401
5402 return BeginStyle(attr);
5403 }
5404
5405 /// Begin using italic
5406 bool wxRichTextBuffer::BeginItalic()
5407 {
5408 wxTextAttr attr;
5409 attr.SetFontStyle(wxITALIC);
5410
5411 return BeginStyle(attr);
5412 }
5413
5414 /// Begin using underline
5415 bool wxRichTextBuffer::BeginUnderline()
5416 {
5417 wxTextAttr attr;
5418 attr.SetFontUnderlined(true);
5419
5420 return BeginStyle(attr);
5421 }
5422
5423 /// Begin using point size
5424 bool wxRichTextBuffer::BeginFontSize(int pointSize)
5425 {
5426 wxTextAttr attr;
5427 attr.SetFontSize(pointSize);
5428
5429 return BeginStyle(attr);
5430 }
5431
5432 /// Begin using this font
5433 bool wxRichTextBuffer::BeginFont(const wxFont& font)
5434 {
5435 wxTextAttr attr;
5436 attr.SetFont(font);
5437
5438 return BeginStyle(attr);
5439 }
5440
5441 /// Begin using this colour
5442 bool wxRichTextBuffer::BeginTextColour(const wxColour& colour)
5443 {
5444 wxTextAttr attr;
5445 attr.SetFlags(wxTEXT_ATTR_TEXT_COLOUR);
5446 attr.SetTextColour(colour);
5447
5448 return BeginStyle(attr);
5449 }
5450
5451 /// Begin using alignment
5452 bool wxRichTextBuffer::BeginAlignment(wxTextAttrAlignment alignment)
5453 {
5454 wxTextAttr attr;
5455 attr.SetFlags(wxTEXT_ATTR_ALIGNMENT);
5456 attr.SetAlignment(alignment);
5457
5458 return BeginStyle(attr);
5459 }
5460
5461 /// Begin left indent
5462 bool wxRichTextBuffer::BeginLeftIndent(int leftIndent, int leftSubIndent)
5463 {
5464 wxTextAttr attr;
5465 attr.SetFlags(wxTEXT_ATTR_LEFT_INDENT);
5466 attr.SetLeftIndent(leftIndent, leftSubIndent);
5467
5468 return BeginStyle(attr);
5469 }
5470
5471 /// Begin right indent
5472 bool wxRichTextBuffer::BeginRightIndent(int rightIndent)
5473 {
5474 wxTextAttr attr;
5475 attr.SetFlags(wxTEXT_ATTR_RIGHT_INDENT);
5476 attr.SetRightIndent(rightIndent);
5477
5478 return BeginStyle(attr);
5479 }
5480
5481 /// Begin paragraph spacing
5482 bool wxRichTextBuffer::BeginParagraphSpacing(int before, int after)
5483 {
5484 long flags = 0;
5485 if (before != 0)
5486 flags |= wxTEXT_ATTR_PARA_SPACING_BEFORE;
5487 if (after != 0)
5488 flags |= wxTEXT_ATTR_PARA_SPACING_AFTER;
5489
5490 wxTextAttr attr;
5491 attr.SetFlags(flags);
5492 attr.SetParagraphSpacingBefore(before);
5493 attr.SetParagraphSpacingAfter(after);
5494
5495 return BeginStyle(attr);
5496 }
5497
5498 /// Begin line spacing
5499 bool wxRichTextBuffer::BeginLineSpacing(int lineSpacing)
5500 {
5501 wxTextAttr attr;
5502 attr.SetFlags(wxTEXT_ATTR_LINE_SPACING);
5503 attr.SetLineSpacing(lineSpacing);
5504
5505 return BeginStyle(attr);
5506 }
5507
5508 /// Begin numbered bullet
5509 bool wxRichTextBuffer::BeginNumberedBullet(int bulletNumber, int leftIndent, int leftSubIndent, int bulletStyle)
5510 {
5511 wxTextAttr attr;
5512 attr.SetFlags(wxTEXT_ATTR_BULLET_STYLE|wxTEXT_ATTR_LEFT_INDENT);
5513 attr.SetBulletStyle(bulletStyle);
5514 attr.SetBulletNumber(bulletNumber);
5515 attr.SetLeftIndent(leftIndent, leftSubIndent);
5516
5517 return BeginStyle(attr);
5518 }
5519
5520 /// Begin symbol bullet
5521 bool wxRichTextBuffer::BeginSymbolBullet(const wxString& symbol, int leftIndent, int leftSubIndent, int bulletStyle)
5522 {
5523 wxTextAttr attr;
5524 attr.SetFlags(wxTEXT_ATTR_BULLET_STYLE|wxTEXT_ATTR_LEFT_INDENT);
5525 attr.SetBulletStyle(bulletStyle);
5526 attr.SetLeftIndent(leftIndent, leftSubIndent);
5527 attr.SetBulletText(symbol);
5528
5529 return BeginStyle(attr);
5530 }
5531
5532 /// Begin standard bullet
5533 bool wxRichTextBuffer::BeginStandardBullet(const wxString& bulletName, int leftIndent, int leftSubIndent, int bulletStyle)
5534 {
5535 wxTextAttr attr;
5536 attr.SetFlags(wxTEXT_ATTR_BULLET_STYLE|wxTEXT_ATTR_LEFT_INDENT);
5537 attr.SetBulletStyle(bulletStyle);
5538 attr.SetLeftIndent(leftIndent, leftSubIndent);
5539 attr.SetBulletName(bulletName);
5540
5541 return BeginStyle(attr);
5542 }
5543
5544 /// Begin named character style
5545 bool wxRichTextBuffer::BeginCharacterStyle(const wxString& characterStyle)
5546 {
5547 if (GetStyleSheet())
5548 {
5549 wxRichTextCharacterStyleDefinition* def = GetStyleSheet()->FindCharacterStyle(characterStyle);
5550 if (def)
5551 {
5552 wxTextAttr attr = def->GetStyleMergedWithBase(GetStyleSheet());
5553 return BeginStyle(attr);
5554 }
5555 }
5556 return false;
5557 }
5558
5559 /// Begin named paragraph style
5560 bool wxRichTextBuffer::BeginParagraphStyle(const wxString& paragraphStyle)
5561 {
5562 if (GetStyleSheet())
5563 {
5564 wxRichTextParagraphStyleDefinition* def = GetStyleSheet()->FindParagraphStyle(paragraphStyle);
5565 if (def)
5566 {
5567 wxTextAttr attr = def->GetStyleMergedWithBase(GetStyleSheet());
5568 return BeginStyle(attr);
5569 }
5570 }
5571 return false;
5572 }
5573
5574 /// Begin named list style
5575 bool wxRichTextBuffer::BeginListStyle(const wxString& listStyle, int level, int number)
5576 {
5577 if (GetStyleSheet())
5578 {
5579 wxRichTextListStyleDefinition* def = GetStyleSheet()->FindListStyle(listStyle);
5580 if (def)
5581 {
5582 wxTextAttr attr(def->GetCombinedStyleForLevel(level));
5583
5584 attr.SetBulletNumber(number);
5585
5586 return BeginStyle(attr);
5587 }
5588 }
5589 return false;
5590 }
5591
5592 /// Begin URL
5593 bool wxRichTextBuffer::BeginURL(const wxString& url, const wxString& characterStyle)
5594 {
5595 wxTextAttr attr;
5596
5597 if (!characterStyle.IsEmpty() && GetStyleSheet())
5598 {
5599 wxRichTextCharacterStyleDefinition* def = GetStyleSheet()->FindCharacterStyle(characterStyle);
5600 if (def)
5601 {
5602 attr = def->GetStyleMergedWithBase(GetStyleSheet());
5603 }
5604 }
5605 attr.SetURL(url);
5606
5607 return BeginStyle(attr);
5608 }
5609
5610 /// Adds a handler to the end
5611 void wxRichTextBuffer::AddHandler(wxRichTextFileHandler *handler)
5612 {
5613 sm_handlers.Append(handler);
5614 }
5615
5616 /// Inserts a handler at the front
5617 void wxRichTextBuffer::InsertHandler(wxRichTextFileHandler *handler)
5618 {
5619 sm_handlers.Insert( handler );
5620 }
5621
5622 /// Removes a handler
5623 bool wxRichTextBuffer::RemoveHandler(const wxString& name)
5624 {
5625 wxRichTextFileHandler *handler = FindHandler(name);
5626 if (handler)
5627 {
5628 sm_handlers.DeleteObject(handler);
5629 delete handler;
5630 return true;
5631 }
5632 else
5633 return false;
5634 }
5635
5636 /// Finds a handler by filename or, if supplied, type
5637 wxRichTextFileHandler *wxRichTextBuffer::FindHandlerFilenameOrType(const wxString& filename, int imageType)
5638 {
5639 if (imageType != wxRICHTEXT_TYPE_ANY)
5640 return FindHandler(imageType);
5641 else if (!filename.IsEmpty())
5642 {
5643 wxString path, file, ext;
5644 wxSplitPath(filename, & path, & file, & ext);
5645 return FindHandler(ext, imageType);
5646 }
5647 else
5648 return NULL;
5649 }
5650
5651
5652 /// Finds a handler by name
5653 wxRichTextFileHandler* wxRichTextBuffer::FindHandler(const wxString& name)
5654 {
5655 wxList::compatibility_iterator node = sm_handlers.GetFirst();
5656 while (node)
5657 {
5658 wxRichTextFileHandler *handler = (wxRichTextFileHandler*)node->GetData();
5659 if (handler->GetName().Lower() == name.Lower()) return handler;
5660
5661 node = node->GetNext();
5662 }
5663 return NULL;
5664 }
5665
5666 /// Finds a handler by extension and type
5667 wxRichTextFileHandler* wxRichTextBuffer::FindHandler(const wxString& extension, int type)
5668 {
5669 wxList::compatibility_iterator node = sm_handlers.GetFirst();
5670 while (node)
5671 {
5672 wxRichTextFileHandler *handler = (wxRichTextFileHandler*)node->GetData();
5673 if ( handler->GetExtension().Lower() == extension.Lower() &&
5674 (type == wxRICHTEXT_TYPE_ANY || handler->GetType() == type) )
5675 return handler;
5676 node = node->GetNext();
5677 }
5678 return 0;
5679 }
5680
5681 /// Finds a handler by type
5682 wxRichTextFileHandler* wxRichTextBuffer::FindHandler(int type)
5683 {
5684 wxList::compatibility_iterator node = sm_handlers.GetFirst();
5685 while (node)
5686 {
5687 wxRichTextFileHandler *handler = (wxRichTextFileHandler *)node->GetData();
5688 if (handler->GetType() == type) return handler;
5689 node = node->GetNext();
5690 }
5691 return NULL;
5692 }
5693
5694 void wxRichTextBuffer::InitStandardHandlers()
5695 {
5696 if (!FindHandler(wxRICHTEXT_TYPE_TEXT))
5697 AddHandler(new wxRichTextPlainTextHandler);
5698 }
5699
5700 void wxRichTextBuffer::CleanUpHandlers()
5701 {
5702 wxList::compatibility_iterator node = sm_handlers.GetFirst();
5703 while (node)
5704 {
5705 wxRichTextFileHandler* handler = (wxRichTextFileHandler*)node->GetData();
5706 wxList::compatibility_iterator next = node->GetNext();
5707 delete handler;
5708 node = next;
5709 }
5710
5711 sm_handlers.Clear();
5712 }
5713
5714 wxString wxRichTextBuffer::GetExtWildcard(bool combine, bool save, wxArrayInt* types)
5715 {
5716 if (types)
5717 types->Clear();
5718
5719 wxString wildcard;
5720
5721 wxList::compatibility_iterator node = GetHandlers().GetFirst();
5722 int count = 0;
5723 while (node)
5724 {
5725 wxRichTextFileHandler* handler = (wxRichTextFileHandler*) node->GetData();
5726 if (handler->IsVisible() && ((save && handler->CanSave()) || !save && handler->CanLoad()))
5727 {
5728 if (combine)
5729 {
5730 if (count > 0)
5731 wildcard += wxT(";");
5732 wildcard += wxT("*.") + handler->GetExtension();
5733 }
5734 else
5735 {
5736 if (count > 0)
5737 wildcard += wxT("|");
5738 wildcard += handler->GetName();
5739 wildcard += wxT(" ");
5740 wildcard += _("files");
5741 wildcard += wxT(" (*.");
5742 wildcard += handler->GetExtension();
5743 wildcard += wxT(")|*.");
5744 wildcard += handler->GetExtension();
5745 if (types)
5746 types->Add(handler->GetType());
5747 }
5748 count ++;
5749 }
5750
5751 node = node->GetNext();
5752 }
5753
5754 if (combine)
5755 wildcard = wxT("(") + wildcard + wxT(")|") + wildcard;
5756 return wildcard;
5757 }
5758
5759 /// Load a file
5760 bool wxRichTextBuffer::LoadFile(const wxString& filename, int type)
5761 {
5762 wxRichTextFileHandler* handler = FindHandlerFilenameOrType(filename, type);
5763 if (handler)
5764 {
5765 SetDefaultStyle(wxTextAttr());
5766 handler->SetFlags(GetHandlerFlags());
5767 bool success = handler->LoadFile(this, filename);
5768 Invalidate(wxRICHTEXT_ALL);
5769 return success;
5770 }
5771 else
5772 return false;
5773 }
5774
5775 /// Save a file
5776 bool wxRichTextBuffer::SaveFile(const wxString& filename, int type)
5777 {
5778 wxRichTextFileHandler* handler = FindHandlerFilenameOrType(filename, type);
5779 if (handler)
5780 {
5781 handler->SetFlags(GetHandlerFlags());
5782 return handler->SaveFile(this, filename);
5783 }
5784 else
5785 return false;
5786 }
5787
5788 /// Load from a stream
5789 bool wxRichTextBuffer::LoadFile(wxInputStream& stream, int type)
5790 {
5791 wxRichTextFileHandler* handler = FindHandler(type);
5792 if (handler)
5793 {
5794 SetDefaultStyle(wxTextAttr());
5795 handler->SetFlags(GetHandlerFlags());
5796 bool success = handler->LoadFile(this, stream);
5797 Invalidate(wxRICHTEXT_ALL);
5798 return success;
5799 }
5800 else
5801 return false;
5802 }
5803
5804 /// Save to a stream
5805 bool wxRichTextBuffer::SaveFile(wxOutputStream& stream, int type)
5806 {
5807 wxRichTextFileHandler* handler = FindHandler(type);
5808 if (handler)
5809 {
5810 handler->SetFlags(GetHandlerFlags());
5811 return handler->SaveFile(this, stream);
5812 }
5813 else
5814 return false;
5815 }
5816
5817 /// Copy the range to the clipboard
5818 bool wxRichTextBuffer::CopyToClipboard(const wxRichTextRange& range)
5819 {
5820 bool success = false;
5821 #if wxUSE_CLIPBOARD && wxUSE_DATAOBJ
5822
5823 if (!wxTheClipboard->IsOpened() && wxTheClipboard->Open())
5824 {
5825 wxTheClipboard->Clear();
5826
5827 // Add composite object
5828
5829 wxDataObjectComposite* compositeObject = new wxDataObjectComposite();
5830
5831 {
5832 wxString text = GetTextForRange(range);
5833
5834 #ifdef __WXMSW__
5835 text = wxTextFile::Translate(text, wxTextFileType_Dos);
5836 #endif
5837
5838 compositeObject->Add(new wxTextDataObject(text), false /* not preferred */);
5839 }
5840
5841 // Add rich text buffer data object. This needs the XML handler to be present.
5842
5843 if (FindHandler(wxRICHTEXT_TYPE_XML))
5844 {
5845 wxRichTextBuffer* richTextBuf = new wxRichTextBuffer;
5846 CopyFragment(range, *richTextBuf);
5847
5848 compositeObject->Add(new wxRichTextBufferDataObject(richTextBuf), true /* preferred */);
5849 }
5850
5851 if (wxTheClipboard->SetData(compositeObject))
5852 success = true;
5853
5854 wxTheClipboard->Close();
5855 }
5856
5857 #else
5858 wxUnusedVar(range);
5859 #endif
5860 return success;
5861 }
5862
5863 /// Paste the clipboard content to the buffer
5864 bool wxRichTextBuffer::PasteFromClipboard(long position)
5865 {
5866 bool success = false;
5867 #if wxUSE_CLIPBOARD && wxUSE_DATAOBJ
5868 if (CanPasteFromClipboard())
5869 {
5870 if (wxTheClipboard->Open())
5871 {
5872 if (wxTheClipboard->IsSupported(wxDataFormat(wxRichTextBufferDataObject::GetRichTextBufferFormatId())))
5873 {
5874 wxRichTextBufferDataObject data;
5875 wxTheClipboard->GetData(data);
5876 wxRichTextBuffer* richTextBuffer = data.GetRichTextBuffer();
5877 if (richTextBuffer)
5878 {
5879 InsertParagraphsWithUndo(position+1, *richTextBuffer, GetRichTextCtrl(), wxRICHTEXT_INSERT_WITH_PREVIOUS_PARAGRAPH_STYLE);
5880 if (GetRichTextCtrl())
5881 GetRichTextCtrl()->ShowPosition(position + richTextBuffer->GetRange().GetEnd());
5882 delete richTextBuffer;
5883 }
5884 }
5885 else if (wxTheClipboard->IsSupported(wxDF_TEXT) || wxTheClipboard->IsSupported(wxDF_UNICODETEXT))
5886 {
5887 wxTextDataObject data;
5888 wxTheClipboard->GetData(data);
5889 wxString text(data.GetText());
5890 #ifdef __WXMSW__
5891 wxString text2;
5892 text2.Alloc(text.Length()+1);
5893 size_t i;
5894 for (i = 0; i < text.Length(); i++)
5895 {
5896 wxChar ch = text[i];
5897 if (ch != wxT('\r'))
5898 text2 += ch;
5899 }
5900 #else
5901 wxString text2 = text;
5902 #endif
5903 InsertTextWithUndo(position+1, text2, GetRichTextCtrl());
5904
5905 if (GetRichTextCtrl())
5906 GetRichTextCtrl()->ShowPosition(position + text2.Length());
5907
5908 success = true;
5909 }
5910 else if (wxTheClipboard->IsSupported(wxDF_BITMAP))
5911 {
5912 wxBitmapDataObject data;
5913 wxTheClipboard->GetData(data);
5914 wxBitmap bitmap(data.GetBitmap());
5915 wxImage image(bitmap.ConvertToImage());
5916
5917 wxRichTextAction* action = new wxRichTextAction(NULL, _("Insert Image"), wxRICHTEXT_INSERT, this, GetRichTextCtrl(), false);
5918
5919 action->GetNewParagraphs().AddImage(image);
5920
5921 if (action->GetNewParagraphs().GetChildCount() == 1)
5922 action->GetNewParagraphs().SetPartialParagraph(true);
5923
5924 action->SetPosition(position);
5925
5926 // Set the range we'll need to delete in Undo
5927 action->SetRange(wxRichTextRange(position, position));
5928
5929 SubmitAction(action);
5930
5931 success = true;
5932 }
5933 wxTheClipboard->Close();
5934 }
5935 }
5936 #else
5937 wxUnusedVar(position);
5938 #endif
5939 return success;
5940 }
5941
5942 /// Can we paste from the clipboard?
5943 bool wxRichTextBuffer::CanPasteFromClipboard() const
5944 {
5945 bool canPaste = false;
5946 #if wxUSE_CLIPBOARD && wxUSE_DATAOBJ
5947 if (!wxTheClipboard->IsOpened() && wxTheClipboard->Open())
5948 {
5949 if (wxTheClipboard->IsSupported(wxDF_TEXT) || wxTheClipboard->IsSupported(wxDF_UNICODETEXT) ||
5950 wxTheClipboard->IsSupported(wxDataFormat(wxRichTextBufferDataObject::GetRichTextBufferFormatId())) ||
5951 wxTheClipboard->IsSupported(wxDF_BITMAP))
5952 {
5953 canPaste = true;
5954 }
5955 wxTheClipboard->Close();
5956 }
5957 #endif
5958 return canPaste;
5959 }
5960
5961 /// Dumps contents of buffer for debugging purposes
5962 void wxRichTextBuffer::Dump()
5963 {
5964 wxString text;
5965 {
5966 wxStringOutputStream stream(& text);
5967 wxTextOutputStream textStream(stream);
5968 Dump(textStream);
5969 }
5970
5971 wxLogDebug(text);
5972 }
5973
5974 /// Add an event handler
5975 bool wxRichTextBuffer::AddEventHandler(wxEvtHandler* handler)
5976 {
5977 m_eventHandlers.Append(handler);
5978 return true;
5979 }
5980
5981 /// Remove an event handler
5982 bool wxRichTextBuffer::RemoveEventHandler(wxEvtHandler* handler, bool deleteHandler)
5983 {
5984 wxList::compatibility_iterator node = m_eventHandlers.Find(handler);
5985 if (node)
5986 {
5987 m_eventHandlers.Erase(node);
5988 if (deleteHandler)
5989 delete handler;
5990
5991 return true;
5992 }
5993 else
5994 return false;
5995 }
5996
5997 /// Clear event handlers
5998 void wxRichTextBuffer::ClearEventHandlers()
5999 {
6000 m_eventHandlers.Clear();
6001 }
6002
6003 /// Send event to event handlers. If sendToAll is true, will send to all event handlers,
6004 /// otherwise will stop at the first successful one.
6005 bool wxRichTextBuffer::SendEvent(wxEvent& event, bool sendToAll)
6006 {
6007 bool success = false;
6008 for (wxList::compatibility_iterator node = m_eventHandlers.GetFirst(); node; node = node->GetNext())
6009 {
6010 wxEvtHandler* handler = (wxEvtHandler*) node->GetData();
6011 if (handler->ProcessEvent(event))
6012 {
6013 success = true;
6014 if (!sendToAll)
6015 return true;
6016 }
6017 }
6018 return success;
6019 }
6020
6021 /// Set style sheet and notify of the change
6022 bool wxRichTextBuffer::SetStyleSheetAndNotify(wxRichTextStyleSheet* sheet)
6023 {
6024 wxRichTextStyleSheet* oldSheet = GetStyleSheet();
6025
6026 wxWindowID id = wxID_ANY;
6027 if (GetRichTextCtrl())
6028 id = GetRichTextCtrl()->GetId();
6029
6030 wxRichTextEvent event(wxEVT_COMMAND_RICHTEXT_STYLESHEET_REPLACING, id);
6031 event.SetEventObject(GetRichTextCtrl());
6032 event.SetOldStyleSheet(oldSheet);
6033 event.SetNewStyleSheet(sheet);
6034 event.Allow();
6035
6036 if (SendEvent(event) && !event.IsAllowed())
6037 {
6038 if (sheet != oldSheet)
6039 delete sheet;
6040
6041 return false;
6042 }
6043
6044 if (oldSheet && oldSheet != sheet)
6045 delete oldSheet;
6046
6047 SetStyleSheet(sheet);
6048
6049 event.SetEventType(wxEVT_COMMAND_RICHTEXT_STYLESHEET_REPLACED);
6050 event.SetOldStyleSheet(NULL);
6051 event.Allow();
6052
6053 return SendEvent(event);
6054 }
6055
6056 /// Set renderer, deleting old one
6057 void wxRichTextBuffer::SetRenderer(wxRichTextRenderer* renderer)
6058 {
6059 if (sm_renderer)
6060 delete sm_renderer;
6061 sm_renderer = renderer;
6062 }
6063
6064 bool wxRichTextStdRenderer::DrawStandardBullet(wxRichTextParagraph* paragraph, wxDC& dc, const wxTextAttr& bulletAttr, const wxRect& rect)
6065 {
6066 if (bulletAttr.GetTextColour().Ok())
6067 {
6068 wxCheckSetPen(dc, wxPen(bulletAttr.GetTextColour()));
6069 wxCheckSetBrush(dc, wxBrush(bulletAttr.GetTextColour()));
6070 }
6071 else
6072 {
6073 wxCheckSetPen(dc, *wxBLACK_PEN);
6074 wxCheckSetBrush(dc, *wxBLACK_BRUSH);
6075 }
6076
6077 wxFont font;
6078 if (bulletAttr.HasFont())
6079 {
6080 font = paragraph->GetBuffer()->GetFontTable().FindFont(bulletAttr);
6081 }
6082 else
6083 font = (*wxNORMAL_FONT);
6084
6085 wxCheckSetFont(dc, font);
6086
6087 int charHeight = dc.GetCharHeight();
6088
6089 int bulletWidth = (int) (((float) charHeight) * wxRichTextBuffer::GetBulletProportion());
6090 int bulletHeight = bulletWidth;
6091
6092 int x = rect.x;
6093
6094 // Calculate the top position of the character (as opposed to the whole line height)
6095 int y = rect.y + (rect.height - charHeight);
6096
6097 // Calculate where the bullet should be positioned
6098 y = y + (charHeight+1)/2 - (bulletHeight+1)/2;
6099
6100 // The margin between a bullet and text.
6101 int margin = paragraph->ConvertTenthsMMToPixels(dc, wxRichTextBuffer::GetBulletRightMargin());
6102
6103 if (bulletAttr.GetBulletStyle() & wxTEXT_ATTR_BULLET_STYLE_ALIGN_RIGHT)
6104 x = rect.x + rect.width - bulletWidth - margin;
6105 else if (bulletAttr.GetBulletStyle() & wxTEXT_ATTR_BULLET_STYLE_ALIGN_CENTRE)
6106 x = x + (rect.width)/2 - bulletWidth/2;
6107
6108 if (bulletAttr.GetBulletName() == wxT("standard/square"))
6109 {
6110 dc.DrawRectangle(x, y, bulletWidth, bulletHeight);
6111 }
6112 else if (bulletAttr.GetBulletName() == wxT("standard/diamond"))
6113 {
6114 wxPoint pts[5];
6115 pts[0].x = x; pts[0].y = y + bulletHeight/2;
6116 pts[1].x = x + bulletWidth/2; pts[1].y = y;
6117 pts[2].x = x + bulletWidth; pts[2].y = y + bulletHeight/2;
6118 pts[3].x = x + bulletWidth/2; pts[3].y = y + bulletHeight;
6119
6120 dc.DrawPolygon(4, pts);
6121 }
6122 else if (bulletAttr.GetBulletName() == wxT("standard/triangle"))
6123 {
6124 wxPoint pts[3];
6125 pts[0].x = x; pts[0].y = y;
6126 pts[1].x = x + bulletWidth; pts[1].y = y + bulletHeight/2;
6127 pts[2].x = x; pts[2].y = y + bulletHeight;
6128
6129 dc.DrawPolygon(3, pts);
6130 }
6131 else // "standard/circle", and catch-all
6132 {
6133 dc.DrawEllipse(x, y, bulletWidth, bulletHeight);
6134 }
6135
6136 return true;
6137 }
6138
6139 bool wxRichTextStdRenderer::DrawTextBullet(wxRichTextParagraph* paragraph, wxDC& dc, const wxTextAttr& attr, const wxRect& rect, const wxString& text)
6140 {
6141 if (!text.empty())
6142 {
6143 wxFont font;
6144 if ((attr.GetBulletStyle() & wxTEXT_ATTR_BULLET_STYLE_SYMBOL) && !attr.GetBulletFont().IsEmpty() && attr.HasFont())
6145 {
6146 wxTextAttr fontAttr;
6147 fontAttr.SetFontSize(attr.GetFontSize());
6148 fontAttr.SetFontStyle(attr.GetFontStyle());
6149 fontAttr.SetFontWeight(attr.GetFontWeight());
6150 fontAttr.SetFontUnderlined(attr.GetFontUnderlined());
6151 fontAttr.SetFontFaceName(attr.GetBulletFont());
6152 font = paragraph->GetBuffer()->GetFontTable().FindFont(fontAttr);
6153 }
6154 else if (attr.HasFont())
6155 font = paragraph->GetBuffer()->GetFontTable().FindFont(attr);
6156 else
6157 font = (*wxNORMAL_FONT);
6158
6159 wxCheckSetFont(dc, font);
6160
6161 if (attr.GetTextColour().Ok())
6162 dc.SetTextForeground(attr.GetTextColour());
6163
6164 dc.SetBackgroundMode(wxBRUSHSTYLE_TRANSPARENT);
6165
6166 int charHeight = dc.GetCharHeight();
6167 wxCoord tw, th;
6168 dc.GetTextExtent(text, & tw, & th);
6169
6170 int x = rect.x;
6171
6172 // Calculate the top position of the character (as opposed to the whole line height)
6173 int y = rect.y + (rect.height - charHeight);
6174
6175 // The margin between a bullet and text.
6176 int margin = paragraph->ConvertTenthsMMToPixels(dc, wxRichTextBuffer::GetBulletRightMargin());
6177
6178 if (attr.GetBulletStyle() & wxTEXT_ATTR_BULLET_STYLE_ALIGN_RIGHT)
6179 x = (rect.x + rect.width) - tw - margin;
6180 else if (attr.GetBulletStyle() & wxTEXT_ATTR_BULLET_STYLE_ALIGN_CENTRE)
6181 x = x + (rect.width)/2 - tw/2;
6182
6183 dc.DrawText(text, x, y);
6184
6185 return true;
6186 }
6187 else
6188 return false;
6189 }
6190
6191 bool wxRichTextStdRenderer::DrawBitmapBullet(wxRichTextParagraph* WXUNUSED(paragraph), wxDC& WXUNUSED(dc), const wxTextAttr& WXUNUSED(attr), const wxRect& WXUNUSED(rect))
6192 {
6193 // Currently unimplemented. The intention is to store bitmaps by name in a media store associated
6194 // with the buffer. The store will allow retrieval from memory, disk or other means.
6195 return false;
6196 }
6197
6198 /// Enumerate the standard bullet names currently supported
6199 bool wxRichTextStdRenderer::EnumerateStandardBulletNames(wxArrayString& bulletNames)
6200 {
6201 bulletNames.Add(wxT("standard/circle"));
6202 bulletNames.Add(wxT("standard/square"));
6203 bulletNames.Add(wxT("standard/diamond"));
6204 bulletNames.Add(wxT("standard/triangle"));
6205
6206 return true;
6207 }
6208
6209 /*
6210 * Module to initialise and clean up handlers
6211 */
6212
6213 class wxRichTextModule: public wxModule
6214 {
6215 DECLARE_DYNAMIC_CLASS(wxRichTextModule)
6216 public:
6217 wxRichTextModule() {}
6218 bool OnInit()
6219 {
6220 wxRichTextBuffer::SetRenderer(new wxRichTextStdRenderer);
6221 wxRichTextBuffer::InitStandardHandlers();
6222 wxRichTextParagraph::InitDefaultTabs();
6223 return true;
6224 }
6225 void OnExit()
6226 {
6227 wxRichTextBuffer::CleanUpHandlers();
6228 wxRichTextDecimalToRoman(-1);
6229 wxRichTextParagraph::ClearDefaultTabs();
6230 wxRichTextCtrl::ClearAvailableFontNames();
6231 wxRichTextBuffer::SetRenderer(NULL);
6232 }
6233 };
6234
6235 IMPLEMENT_DYNAMIC_CLASS(wxRichTextModule, wxModule)
6236
6237
6238 // If the richtext lib is dynamically loaded after the app has already started
6239 // (such as from wxPython) then the built-in module system will not init this
6240 // module. Provide this function to do it manually.
6241 void wxRichTextModuleInit()
6242 {
6243 wxModule* module = new wxRichTextModule;
6244 module->Init();
6245 wxModule::RegisterModule(module);
6246 }
6247
6248
6249 /*!
6250 * Commands for undo/redo
6251 *
6252 */
6253
6254 wxRichTextCommand::wxRichTextCommand(const wxString& name, wxRichTextCommandId id, wxRichTextBuffer* buffer,
6255 wxRichTextCtrl* ctrl, bool ignoreFirstTime): wxCommand(true, name)
6256 {
6257 /* wxRichTextAction* action = */ new wxRichTextAction(this, name, id, buffer, ctrl, ignoreFirstTime);
6258 }
6259
6260 wxRichTextCommand::wxRichTextCommand(const wxString& name): wxCommand(true, name)
6261 {
6262 }
6263
6264 wxRichTextCommand::~wxRichTextCommand()
6265 {
6266 ClearActions();
6267 }
6268
6269 void wxRichTextCommand::AddAction(wxRichTextAction* action)
6270 {
6271 if (!m_actions.Member(action))
6272 m_actions.Append(action);
6273 }
6274
6275 bool wxRichTextCommand::Do()
6276 {
6277 for (wxList::compatibility_iterator node = m_actions.GetFirst(); node; node = node->GetNext())
6278 {
6279 wxRichTextAction* action = (wxRichTextAction*) node->GetData();
6280 action->Do();
6281 }
6282
6283 return true;
6284 }
6285
6286 bool wxRichTextCommand::Undo()
6287 {
6288 for (wxList::compatibility_iterator node = m_actions.GetLast(); node; node = node->GetPrevious())
6289 {
6290 wxRichTextAction* action = (wxRichTextAction*) node->GetData();
6291 action->Undo();
6292 }
6293
6294 return true;
6295 }
6296
6297 void wxRichTextCommand::ClearActions()
6298 {
6299 WX_CLEAR_LIST(wxList, m_actions);
6300 }
6301
6302 /*!
6303 * Individual action
6304 *
6305 */
6306
6307 wxRichTextAction::wxRichTextAction(wxRichTextCommand* cmd, const wxString& name, wxRichTextCommandId id, wxRichTextBuffer* buffer,
6308 wxRichTextCtrl* ctrl, bool ignoreFirstTime)
6309 {
6310 m_buffer = buffer;
6311 m_ignoreThis = ignoreFirstTime;
6312 m_cmdId = id;
6313 m_position = -1;
6314 m_ctrl = ctrl;
6315 m_name = name;
6316 m_newParagraphs.SetDefaultStyle(buffer->GetDefaultStyle());
6317 m_newParagraphs.SetBasicStyle(buffer->GetBasicStyle());
6318 if (cmd)
6319 cmd->AddAction(this);
6320 }
6321
6322 wxRichTextAction::~wxRichTextAction()
6323 {
6324 }
6325
6326 bool wxRichTextAction::Do()
6327 {
6328 m_buffer->Modify(true);
6329
6330 switch (m_cmdId)
6331 {
6332 case wxRICHTEXT_INSERT:
6333 {
6334 // Store a list of line start character and y positions so we can figure out which area
6335 // we need to refresh
6336 wxArrayInt optimizationLineCharPositions;
6337 wxArrayInt optimizationLineYPositions;
6338
6339 #if wxRICHTEXT_USE_OPTIMIZED_DRAWING
6340 // NOTE: we're assuming that the buffer is laid out correctly at this point.
6341 // If we had several actions, which only invalidate and leave layout until the
6342 // paint handler is called, then this might not be true. So we may need to switch
6343 // optimisation on only when we're simply adding text and not simultaneously
6344 // deleting a selection, for example. Or, we make sure the buffer is laid out correctly
6345 // first, but of course this means we'll be doing it twice.
6346 if (!m_buffer->GetDirty() && m_ctrl) // can only do optimisation if the buffer is already laid out correctly
6347 {
6348 wxSize clientSize = m_ctrl->GetClientSize();
6349 wxPoint firstVisiblePt = m_ctrl->GetFirstVisiblePoint();
6350 int lastY = firstVisiblePt.y + clientSize.y;
6351
6352 wxRichTextParagraph* para = m_buffer->GetParagraphAtPosition(GetRange().GetStart());
6353 wxRichTextObjectList::compatibility_iterator node = m_buffer->GetChildren().Find(para);
6354 while (node)
6355 {
6356 wxRichTextParagraph* child = (wxRichTextParagraph*) node->GetData();
6357 wxRichTextLineList::compatibility_iterator node2 = child->GetLines().GetFirst();
6358 while (node2)
6359 {
6360 wxRichTextLine* line = node2->GetData();
6361 wxPoint pt = line->GetAbsolutePosition();
6362 wxRichTextRange range = line->GetAbsoluteRange();
6363
6364 if (pt.y > lastY)
6365 {
6366 node2 = wxRichTextLineList::compatibility_iterator();
6367 node = wxRichTextObjectList::compatibility_iterator();
6368 }
6369 else if (range.GetStart() > GetPosition() && pt.y >= firstVisiblePt.y)
6370 {
6371 optimizationLineCharPositions.Add(range.GetStart());
6372 optimizationLineYPositions.Add(pt.y);
6373 }
6374
6375 if (node2)
6376 node2 = node2->GetNext();
6377 }
6378
6379 if (node)
6380 node = node->GetNext();
6381 }
6382 }
6383 #endif
6384
6385 m_buffer->InsertFragment(GetRange().GetStart(), m_newParagraphs);
6386 m_buffer->UpdateRanges();
6387 m_buffer->Invalidate(wxRichTextRange(wxMax(0, GetRange().GetStart()-1), GetRange().GetEnd()));
6388
6389 long newCaretPosition = GetPosition() + m_newParagraphs.GetRange().GetLength();
6390
6391 // Character position to caret position
6392 newCaretPosition --;
6393
6394 // Don't take into account the last newline
6395 if (m_newParagraphs.GetPartialParagraph())
6396 newCaretPosition --;
6397 else
6398 if (m_newParagraphs.GetChildren().GetCount() > 1)
6399 {
6400 wxRichTextObject* p = (wxRichTextObject*) m_newParagraphs.GetChildren().GetLast()->GetData();
6401 if (p->GetRange().GetLength() == 1)
6402 newCaretPosition --;
6403 }
6404
6405 newCaretPosition = wxMin(newCaretPosition, (m_buffer->GetRange().GetEnd()-1));
6406
6407 if (optimizationLineCharPositions.GetCount() > 0)
6408 UpdateAppearance(newCaretPosition, true /* send update event */, & optimizationLineCharPositions, & optimizationLineYPositions);
6409 else
6410 UpdateAppearance(newCaretPosition, true /* send update event */);
6411
6412 wxRichTextEvent cmdEvent(
6413 wxEVT_COMMAND_RICHTEXT_CONTENT_INSERTED,
6414 m_ctrl ? m_ctrl->GetId() : -1);
6415 cmdEvent.SetEventObject(m_ctrl ? (wxObject*) m_ctrl : (wxObject*) m_buffer);
6416 cmdEvent.SetRange(GetRange());
6417 cmdEvent.SetPosition(GetRange().GetStart());
6418
6419 m_buffer->SendEvent(cmdEvent);
6420
6421 break;
6422 }
6423 case wxRICHTEXT_DELETE:
6424 {
6425 m_buffer->DeleteRange(GetRange());
6426 m_buffer->UpdateRanges();
6427 m_buffer->Invalidate(wxRichTextRange(GetRange().GetStart(), GetRange().GetStart()));
6428
6429 long caretPos = GetRange().GetStart()-1;
6430 if (caretPos >= m_buffer->GetRange().GetEnd())
6431 caretPos --;
6432
6433 UpdateAppearance(caretPos, true /* send update event */);
6434
6435 wxRichTextEvent cmdEvent(
6436 wxEVT_COMMAND_RICHTEXT_CONTENT_DELETED,
6437 m_ctrl ? m_ctrl->GetId() : -1);
6438 cmdEvent.SetEventObject(m_ctrl ? (wxObject*) m_ctrl : (wxObject*) m_buffer);
6439 cmdEvent.SetRange(GetRange());
6440 cmdEvent.SetPosition(GetRange().GetStart());
6441
6442 m_buffer->SendEvent(cmdEvent);
6443
6444 break;
6445 }
6446 case wxRICHTEXT_CHANGE_STYLE:
6447 {
6448 ApplyParagraphs(GetNewParagraphs());
6449 m_buffer->Invalidate(GetRange());
6450
6451 UpdateAppearance(GetPosition());
6452
6453 wxRichTextEvent cmdEvent(
6454 wxEVT_COMMAND_RICHTEXT_STYLE_CHANGED,
6455 m_ctrl ? m_ctrl->GetId() : -1);
6456 cmdEvent.SetEventObject(m_ctrl ? (wxObject*) m_ctrl : (wxObject*) m_buffer);
6457 cmdEvent.SetRange(GetRange());
6458 cmdEvent.SetPosition(GetRange().GetStart());
6459
6460 m_buffer->SendEvent(cmdEvent);
6461
6462 break;
6463 }
6464 default:
6465 break;
6466 }
6467
6468 return true;
6469 }
6470
6471 bool wxRichTextAction::Undo()
6472 {
6473 m_buffer->Modify(true);
6474
6475 switch (m_cmdId)
6476 {
6477 case wxRICHTEXT_INSERT:
6478 {
6479 m_buffer->DeleteRange(GetRange());
6480 m_buffer->UpdateRanges();
6481 m_buffer->Invalidate(wxRichTextRange(GetRange().GetStart(), GetRange().GetStart()));
6482
6483 long newCaretPosition = GetPosition() - 1;
6484
6485 UpdateAppearance(newCaretPosition, true /* send update event */);
6486
6487 wxRichTextEvent cmdEvent(
6488 wxEVT_COMMAND_RICHTEXT_CONTENT_DELETED,
6489 m_ctrl ? m_ctrl->GetId() : -1);
6490 cmdEvent.SetEventObject(m_ctrl ? (wxObject*) m_ctrl : (wxObject*) m_buffer);
6491 cmdEvent.SetRange(GetRange());
6492 cmdEvent.SetPosition(GetRange().GetStart());
6493
6494 m_buffer->SendEvent(cmdEvent);
6495
6496 break;
6497 }
6498 case wxRICHTEXT_DELETE:
6499 {
6500 m_buffer->InsertFragment(GetRange().GetStart(), m_oldParagraphs);
6501 m_buffer->UpdateRanges();
6502 m_buffer->Invalidate(GetRange());
6503
6504 UpdateAppearance(GetPosition(), true /* send update event */);
6505
6506 wxRichTextEvent cmdEvent(
6507 wxEVT_COMMAND_RICHTEXT_CONTENT_INSERTED,
6508 m_ctrl ? m_ctrl->GetId() : -1);
6509 cmdEvent.SetEventObject(m_ctrl ? (wxObject*) m_ctrl : (wxObject*) m_buffer);
6510 cmdEvent.SetRange(GetRange());
6511 cmdEvent.SetPosition(GetRange().GetStart());
6512
6513 m_buffer->SendEvent(cmdEvent);
6514
6515 break;
6516 }
6517 case wxRICHTEXT_CHANGE_STYLE:
6518 {
6519 ApplyParagraphs(GetOldParagraphs());
6520 m_buffer->Invalidate(GetRange());
6521
6522 UpdateAppearance(GetPosition());
6523
6524 wxRichTextEvent cmdEvent(
6525 wxEVT_COMMAND_RICHTEXT_STYLE_CHANGED,
6526 m_ctrl ? m_ctrl->GetId() : -1);
6527 cmdEvent.SetEventObject(m_ctrl ? (wxObject*) m_ctrl : (wxObject*) m_buffer);
6528 cmdEvent.SetRange(GetRange());
6529 cmdEvent.SetPosition(GetRange().GetStart());
6530
6531 m_buffer->SendEvent(cmdEvent);
6532
6533 break;
6534 }
6535 default:
6536 break;
6537 }
6538
6539 return true;
6540 }
6541
6542 /// Update the control appearance
6543 void wxRichTextAction::UpdateAppearance(long caretPosition, bool sendUpdateEvent, wxArrayInt* optimizationLineCharPositions, wxArrayInt* optimizationLineYPositions)
6544 {
6545 if (m_ctrl)
6546 {
6547 m_ctrl->SetCaretPosition(caretPosition);
6548 if (!m_ctrl->IsFrozen())
6549 {
6550 m_ctrl->LayoutContent();
6551 m_ctrl->PositionCaret();
6552
6553 #if wxRICHTEXT_USE_OPTIMIZED_DRAWING
6554 // Find refresh rectangle if we are in a position to optimise refresh
6555 if (m_cmdId == wxRICHTEXT_INSERT && optimizationLineCharPositions && optimizationLineCharPositions->GetCount() > 0)
6556 {
6557 size_t i;
6558
6559 wxSize clientSize = m_ctrl->GetClientSize();
6560 wxPoint firstVisiblePt = m_ctrl->GetFirstVisiblePoint();
6561
6562 // Start/end positions
6563 int firstY = 0;
6564 int lastY = firstVisiblePt.y + clientSize.y;
6565
6566 bool foundStart = false;
6567 bool foundEnd = false;
6568
6569 // position offset - how many characters were inserted
6570 int positionOffset = GetRange().GetLength();
6571
6572 // find the first line which is being drawn at the same position as it was
6573 // before. Since we're talking about a simple insertion, we can assume
6574 // that the rest of the window does not need to be redrawn.
6575
6576 wxRichTextParagraph* para = m_buffer->GetParagraphAtPosition(GetPosition());
6577 wxRichTextObjectList::compatibility_iterator node = m_buffer->GetChildren().Find(para);
6578 while (node)
6579 {
6580 wxRichTextParagraph* child = (wxRichTextParagraph*) node->GetData();
6581 wxRichTextLineList::compatibility_iterator node2 = child->GetLines().GetFirst();
6582 while (node2)
6583 {
6584 wxRichTextLine* line = node2->GetData();
6585 wxPoint pt = line->GetAbsolutePosition();
6586 wxRichTextRange range = line->GetAbsoluteRange();
6587
6588 // we want to find the first line that is in the same position
6589 // as before. This will mean we're at the end of the changed text.
6590
6591 if (pt.y > lastY) // going past the end of the window, no more info
6592 {
6593 node2 = wxRichTextLineList::compatibility_iterator();
6594 node = wxRichTextObjectList::compatibility_iterator();
6595 }
6596 else
6597 {
6598 if (!foundStart)
6599 {
6600 firstY = pt.y - firstVisiblePt.y;
6601 foundStart = true;
6602 }
6603
6604 // search for this line being at the same position as before
6605 for (i = 0; i < optimizationLineCharPositions->GetCount(); i++)
6606 {
6607 if (((*optimizationLineCharPositions)[i] + positionOffset == range.GetStart()) &&
6608 ((*optimizationLineYPositions)[i] == pt.y))
6609 {
6610 // Stop, we're now the same as we were
6611 foundEnd = true;
6612 lastY = pt.y - firstVisiblePt.y;
6613
6614 node2 = wxRichTextLineList::compatibility_iterator();
6615 node = wxRichTextObjectList::compatibility_iterator();
6616
6617 break;
6618 }
6619 }
6620 }
6621
6622 if (node2)
6623 node2 = node2->GetNext();
6624 }
6625
6626 if (node)
6627 node = node->GetNext();
6628 }
6629
6630 if (!foundStart)
6631 firstY = firstVisiblePt.y;
6632 if (!foundEnd)
6633 lastY = firstVisiblePt.y + clientSize.y;
6634
6635 wxRect rect(firstVisiblePt.x, firstY, firstVisiblePt.x + clientSize.x, lastY - firstY);
6636 m_ctrl->RefreshRect(rect);
6637
6638 // TODO: we need to make sure that lines are only drawn if in the update region. The rect
6639 // passed to Draw is currently used in different ways (to pass the position the content should
6640 // be drawn at as well as the relevant region).
6641 }
6642 else
6643 #endif
6644 m_ctrl->Refresh(false);
6645
6646 if (sendUpdateEvent)
6647 wxTextCtrl::SendTextUpdatedEvent(m_ctrl);
6648 }
6649 }
6650 }
6651
6652 /// Replace the buffer paragraphs with the new ones.
6653 void wxRichTextAction::ApplyParagraphs(const wxRichTextParagraphLayoutBox& fragment)
6654 {
6655 wxRichTextObjectList::compatibility_iterator node = fragment.GetChildren().GetFirst();
6656 while (node)
6657 {
6658 wxRichTextParagraph* para = wxDynamicCast(node->GetData(), wxRichTextParagraph);
6659 wxASSERT (para != NULL);
6660
6661 // We'll replace the existing paragraph by finding the paragraph at this position,
6662 // delete its node data, and setting a copy as the new node data.
6663 // TODO: make more efficient by simply swapping old and new paragraph objects.
6664
6665 wxRichTextParagraph* existingPara = m_buffer->GetParagraphAtPosition(para->GetRange().GetStart());
6666 if (existingPara)
6667 {
6668 wxRichTextObjectList::compatibility_iterator bufferParaNode = m_buffer->GetChildren().Find(existingPara);
6669 if (bufferParaNode)
6670 {
6671 wxRichTextParagraph* newPara = new wxRichTextParagraph(*para);
6672 newPara->SetParent(m_buffer);
6673
6674 bufferParaNode->SetData(newPara);
6675
6676 delete existingPara;
6677 }
6678 }
6679
6680 node = node->GetNext();
6681 }
6682 }
6683
6684
6685 /*!
6686 * wxRichTextRange
6687 * This stores beginning and end positions for a range of data.
6688 */
6689
6690 /// Limit this range to be within 'range'
6691 bool wxRichTextRange::LimitTo(const wxRichTextRange& range)
6692 {
6693 if (m_start < range.m_start)
6694 m_start = range.m_start;
6695
6696 if (m_end > range.m_end)
6697 m_end = range.m_end;
6698
6699 return true;
6700 }
6701
6702 /*!
6703 * wxRichTextImage implementation
6704 * This object represents an image.
6705 */
6706
6707 IMPLEMENT_DYNAMIC_CLASS(wxRichTextImage, wxRichTextObject)
6708
6709 wxRichTextImage::wxRichTextImage(const wxImage& image, wxRichTextObject* parent, wxTextAttr* charStyle):
6710 wxRichTextObject(parent)
6711 {
6712 m_image = image;
6713 if (charStyle)
6714 SetAttributes(*charStyle);
6715 }
6716
6717 wxRichTextImage::wxRichTextImage(const wxRichTextImageBlock& imageBlock, wxRichTextObject* parent, wxTextAttr* charStyle):
6718 wxRichTextObject(parent)
6719 {
6720 m_imageBlock = imageBlock;
6721 m_imageBlock.Load(m_image);
6722 if (charStyle)
6723 SetAttributes(*charStyle);
6724 }
6725
6726 /// Load wxImage from the block
6727 bool wxRichTextImage::LoadFromBlock()
6728 {
6729 m_imageBlock.Load(m_image);
6730 return m_imageBlock.Ok();
6731 }
6732
6733 /// Make block from the wxImage
6734 bool wxRichTextImage::MakeBlock()
6735 {
6736 if (m_imageBlock.GetImageType() == wxBITMAP_TYPE_ANY || m_imageBlock.GetImageType() == -1)
6737 m_imageBlock.SetImageType(wxBITMAP_TYPE_PNG);
6738
6739 m_imageBlock.MakeImageBlock(m_image, m_imageBlock.GetImageType());
6740 return m_imageBlock.Ok();
6741 }
6742
6743
6744 /// Draw the item
6745 bool wxRichTextImage::Draw(wxDC& dc, const wxRichTextRange& range, const wxRichTextRange& selectionRange, const wxRect& rect, int WXUNUSED(descent), int WXUNUSED(style))
6746 {
6747 if (!m_image.Ok() && m_imageBlock.Ok())
6748 LoadFromBlock();
6749
6750 if (!m_image.Ok())
6751 return false;
6752
6753 if (m_image.Ok() && !m_bitmap.Ok())
6754 m_bitmap = wxBitmap(m_image);
6755
6756 int y = rect.y + (rect.height - m_image.GetHeight());
6757
6758 if (m_bitmap.Ok())
6759 dc.DrawBitmap(m_bitmap, rect.x, y, true);
6760
6761 if (selectionRange.Contains(range.GetStart()))
6762 {
6763 wxCheckSetBrush(dc, *wxBLACK_BRUSH);
6764 wxCheckSetPen(dc, *wxBLACK_PEN);
6765 dc.SetLogicalFunction(wxINVERT);
6766 dc.DrawRectangle(rect);
6767 dc.SetLogicalFunction(wxCOPY);
6768 }
6769
6770 return true;
6771 }
6772
6773 /// Lay the item out
6774 bool wxRichTextImage::Layout(wxDC& WXUNUSED(dc), const wxRect& rect, int WXUNUSED(style))
6775 {
6776 if (!m_image.Ok())
6777 LoadFromBlock();
6778
6779 if (m_image.Ok())
6780 {
6781 SetCachedSize(wxSize(m_image.GetWidth(), m_image.GetHeight()));
6782 SetPosition(rect.GetPosition());
6783 }
6784
6785 return true;
6786 }
6787
6788 /// Get/set the object size for the given range. Returns false if the range
6789 /// is invalid for this object.
6790 bool wxRichTextImage::GetRangeSize(const wxRichTextRange& range, wxSize& size, int& WXUNUSED(descent), wxDC& WXUNUSED(dc), int WXUNUSED(flags), wxPoint WXUNUSED(position), wxArrayInt* partialExtents) const
6791 {
6792 if (!range.IsWithin(GetRange()))
6793 return false;
6794
6795 if (partialExtents)
6796 {
6797 if (m_image.Ok())
6798 partialExtents->Add(m_image.GetWidth());
6799 else
6800 partialExtents->Add(0);
6801 }
6802
6803 if (!m_image.Ok())
6804 return false;
6805
6806 size.x = m_image.GetWidth();
6807 size.y = m_image.GetHeight();
6808
6809 return true;
6810 }
6811
6812 /// Copy
6813 void wxRichTextImage::Copy(const wxRichTextImage& obj)
6814 {
6815 wxRichTextObject::Copy(obj);
6816
6817 m_image = obj.m_image;
6818 m_imageBlock = obj.m_imageBlock;
6819 }
6820
6821 /*!
6822 * Utilities
6823 *
6824 */
6825
6826 /// Compare two attribute objects
6827 bool wxTextAttrEq(const wxTextAttr& attr1, const wxTextAttr& attr2)
6828 {
6829 return (attr1 == attr2);
6830 }
6831
6832 // Partial equality test taking flags into account
6833 bool wxTextAttrEqPartial(const wxTextAttr& attr1, const wxTextAttr& attr2, int flags)
6834 {
6835 return attr1.EqPartial(attr2, flags);
6836 }
6837
6838 /// Compare tabs
6839 bool wxRichTextTabsEq(const wxArrayInt& tabs1, const wxArrayInt& tabs2)
6840 {
6841 if (tabs1.GetCount() != tabs2.GetCount())
6842 return false;
6843
6844 size_t i;
6845 for (i = 0; i < tabs1.GetCount(); i++)
6846 {
6847 if (tabs1[i] != tabs2[i])
6848 return false;
6849 }
6850 return true;
6851 }
6852
6853 bool wxRichTextApplyStyle(wxTextAttr& destStyle, const wxTextAttr& style, wxTextAttr* compareWith)
6854 {
6855 return destStyle.Apply(style, compareWith);
6856 }
6857
6858 // Remove attributes
6859 bool wxRichTextRemoveStyle(wxTextAttr& destStyle, const wxTextAttr& style)
6860 {
6861 return wxTextAttr::RemoveStyle(destStyle, style);
6862 }
6863
6864 /// Combine two bitlists, specifying the bits of interest with separate flags.
6865 bool wxRichTextCombineBitlists(int& valueA, int valueB, int& flagsA, int flagsB)
6866 {
6867 return wxTextAttr::CombineBitlists(valueA, valueB, flagsA, flagsB);
6868 }
6869
6870 /// Compare two bitlists
6871 bool wxRichTextBitlistsEqPartial(int valueA, int valueB, int flags)
6872 {
6873 return wxTextAttr::BitlistsEqPartial(valueA, valueB, flags);
6874 }
6875
6876 /// Split into paragraph and character styles
6877 bool wxRichTextSplitParaCharStyles(const wxTextAttr& style, wxTextAttr& parStyle, wxTextAttr& charStyle)
6878 {
6879 return wxTextAttr::SplitParaCharStyles(style, parStyle, charStyle);
6880 }
6881
6882 /// Convert a decimal to Roman numerals
6883 wxString wxRichTextDecimalToRoman(long n)
6884 {
6885 static wxArrayInt decimalNumbers;
6886 static wxArrayString romanNumbers;
6887
6888 // Clean up arrays
6889 if (n == -1)
6890 {
6891 decimalNumbers.Clear();
6892 romanNumbers.Clear();
6893 return wxEmptyString;
6894 }
6895
6896 if (decimalNumbers.GetCount() == 0)
6897 {
6898 #define wxRichTextAddDecRom(n, r) decimalNumbers.Add(n); romanNumbers.Add(r);
6899
6900 wxRichTextAddDecRom(1000, wxT("M"));
6901 wxRichTextAddDecRom(900, wxT("CM"));
6902 wxRichTextAddDecRom(500, wxT("D"));
6903 wxRichTextAddDecRom(400, wxT("CD"));
6904 wxRichTextAddDecRom(100, wxT("C"));
6905 wxRichTextAddDecRom(90, wxT("XC"));
6906 wxRichTextAddDecRom(50, wxT("L"));
6907 wxRichTextAddDecRom(40, wxT("XL"));
6908 wxRichTextAddDecRom(10, wxT("X"));
6909 wxRichTextAddDecRom(9, wxT("IX"));
6910 wxRichTextAddDecRom(5, wxT("V"));
6911 wxRichTextAddDecRom(4, wxT("IV"));
6912 wxRichTextAddDecRom(1, wxT("I"));
6913 }
6914
6915 int i = 0;
6916 wxString roman;
6917
6918 while (n > 0 && i < 13)
6919 {
6920 if (n >= decimalNumbers[i])
6921 {
6922 n -= decimalNumbers[i];
6923 roman += romanNumbers[i];
6924 }
6925 else
6926 {
6927 i ++;
6928 }
6929 }
6930 if (roman.IsEmpty())
6931 roman = wxT("0");
6932 return roman;
6933 }
6934
6935 /*!
6936 * wxRichTextFileHandler
6937 * Base class for file handlers
6938 */
6939
6940 IMPLEMENT_CLASS(wxRichTextFileHandler, wxObject)
6941
6942 #if wxUSE_FFILE && wxUSE_STREAMS
6943 bool wxRichTextFileHandler::LoadFile(wxRichTextBuffer *buffer, const wxString& filename)
6944 {
6945 wxFFileInputStream stream(filename);
6946 if (stream.Ok())
6947 return LoadFile(buffer, stream);
6948
6949 return false;
6950 }
6951
6952 bool wxRichTextFileHandler::SaveFile(wxRichTextBuffer *buffer, const wxString& filename)
6953 {
6954 wxFFileOutputStream stream(filename);
6955 if (stream.Ok())
6956 return SaveFile(buffer, stream);
6957
6958 return false;
6959 }
6960 #endif // wxUSE_FFILE && wxUSE_STREAMS
6961
6962 /// Can we handle this filename (if using files)? By default, checks the extension.
6963 bool wxRichTextFileHandler::CanHandle(const wxString& filename) const
6964 {
6965 wxString path, file, ext;
6966 wxSplitPath(filename, & path, & file, & ext);
6967
6968 return (ext.Lower() == GetExtension());
6969 }
6970
6971 /*!
6972 * wxRichTextTextHandler
6973 * Plain text handler
6974 */
6975
6976 IMPLEMENT_CLASS(wxRichTextPlainTextHandler, wxRichTextFileHandler)
6977
6978 #if wxUSE_STREAMS
6979 bool wxRichTextPlainTextHandler::DoLoadFile(wxRichTextBuffer *buffer, wxInputStream& stream)
6980 {
6981 if (!stream.IsOk())
6982 return false;
6983
6984 wxString str;
6985 int lastCh = 0;
6986
6987 while (!stream.Eof())
6988 {
6989 int ch = stream.GetC();
6990
6991 if (!stream.Eof())
6992 {
6993 if (ch == 10 && lastCh != 13)
6994 str += wxT('\n');
6995
6996 if (ch > 0 && ch != 10)
6997 str += wxChar(ch);
6998
6999 lastCh = ch;
7000 }
7001 }
7002
7003 buffer->ResetAndClearCommands();
7004 buffer->Clear();
7005 buffer->AddParagraphs(str);
7006 buffer->UpdateRanges();
7007
7008 return true;
7009 }
7010
7011 bool wxRichTextPlainTextHandler::DoSaveFile(wxRichTextBuffer *buffer, wxOutputStream& stream)
7012 {
7013 if (!stream.IsOk())
7014 return false;
7015
7016 wxString text = buffer->GetText();
7017
7018 wxString newLine = wxRichTextLineBreakChar;
7019 text.Replace(newLine, wxT("\n"));
7020
7021 wxCharBuffer buf = text.ToAscii();
7022
7023 stream.Write((const char*) buf, text.length());
7024 return true;
7025 }
7026 #endif // wxUSE_STREAMS
7027
7028 /*
7029 * Stores information about an image, in binary in-memory form
7030 */
7031
7032 wxRichTextImageBlock::wxRichTextImageBlock()
7033 {
7034 Init();
7035 }
7036
7037 wxRichTextImageBlock::wxRichTextImageBlock(const wxRichTextImageBlock& block):wxObject()
7038 {
7039 Init();
7040 Copy(block);
7041 }
7042
7043 wxRichTextImageBlock::~wxRichTextImageBlock()
7044 {
7045 if (m_data)
7046 {
7047 delete[] m_data;
7048 m_data = NULL;
7049 }
7050 }
7051
7052 void wxRichTextImageBlock::Init()
7053 {
7054 m_data = NULL;
7055 m_dataSize = 0;
7056 m_imageType = -1;
7057 }
7058
7059 void wxRichTextImageBlock::Clear()
7060 {
7061 delete[] m_data;
7062 m_data = NULL;
7063 m_dataSize = 0;
7064 m_imageType = -1;
7065 }
7066
7067
7068 // Load the original image into a memory block.
7069 // If the image is not a JPEG, we must convert it into a JPEG
7070 // to conserve space.
7071 // If it's not a JPEG we can make use of 'image', already scaled, so we don't have to
7072 // load the image a 2nd time.
7073
7074 bool wxRichTextImageBlock::MakeImageBlock(const wxString& filename, int imageType, wxImage& image, bool convertToJPEG)
7075 {
7076 m_imageType = imageType;
7077
7078 wxString filenameToRead(filename);
7079 bool removeFile = false;
7080
7081 if (imageType == -1)
7082 return false; // Could not determine image type
7083
7084 if ((imageType != wxBITMAP_TYPE_JPEG) && convertToJPEG)
7085 {
7086 wxString tempFile;
7087 bool success = wxGetTempFileName(_("image"), tempFile) ;
7088
7089 wxASSERT(success);
7090
7091 wxUnusedVar(success);
7092
7093 image.SaveFile(tempFile, wxBITMAP_TYPE_JPEG);
7094 filenameToRead = tempFile;
7095 removeFile = true;
7096
7097 m_imageType = wxBITMAP_TYPE_JPEG;
7098 }
7099 wxFile file;
7100 if (!file.Open(filenameToRead))
7101 return false;
7102
7103 m_dataSize = (size_t) file.Length();
7104 file.Close();
7105
7106 if (m_data)
7107 delete[] m_data;
7108 m_data = ReadBlock(filenameToRead, m_dataSize);
7109
7110 if (removeFile)
7111 wxRemoveFile(filenameToRead);
7112
7113 return (m_data != NULL);
7114 }
7115
7116 // Make an image block from the wxImage in the given
7117 // format.
7118 bool wxRichTextImageBlock::MakeImageBlock(wxImage& image, int imageType, int quality)
7119 {
7120 m_imageType = imageType;
7121 image.SetOption(wxT("quality"), quality);
7122
7123 if (imageType == -1)
7124 return false; // Could not determine image type
7125
7126 wxString tempFile;
7127 bool success = wxGetTempFileName(_("image"), tempFile) ;
7128
7129 wxASSERT(success);
7130 wxUnusedVar(success);
7131
7132 if (!image.SaveFile(tempFile, m_imageType))
7133 {
7134 if (wxFileExists(tempFile))
7135 wxRemoveFile(tempFile);
7136 return false;
7137 }
7138
7139 wxFile file;
7140 if (!file.Open(tempFile))
7141 return false;
7142
7143 m_dataSize = (size_t) file.Length();
7144 file.Close();
7145
7146 if (m_data)
7147 delete[] m_data;
7148 m_data = ReadBlock(tempFile, m_dataSize);
7149
7150 wxRemoveFile(tempFile);
7151
7152 return (m_data != NULL);
7153 }
7154
7155
7156 // Write to a file
7157 bool wxRichTextImageBlock::Write(const wxString& filename)
7158 {
7159 return WriteBlock(filename, m_data, m_dataSize);
7160 }
7161
7162 void wxRichTextImageBlock::Copy(const wxRichTextImageBlock& block)
7163 {
7164 m_imageType = block.m_imageType;
7165 if (m_data)
7166 {
7167 delete[] m_data;
7168 m_data = NULL;
7169 }
7170 m_dataSize = block.m_dataSize;
7171 if (m_dataSize == 0)
7172 return;
7173
7174 m_data = new unsigned char[m_dataSize];
7175 unsigned int i;
7176 for (i = 0; i < m_dataSize; i++)
7177 m_data[i] = block.m_data[i];
7178 }
7179
7180 //// Operators
7181 void wxRichTextImageBlock::operator=(const wxRichTextImageBlock& block)
7182 {
7183 Copy(block);
7184 }
7185
7186 // Load a wxImage from the block
7187 bool wxRichTextImageBlock::Load(wxImage& image)
7188 {
7189 if (!m_data)
7190 return false;
7191
7192 // Read in the image.
7193 #if wxUSE_STREAMS
7194 wxMemoryInputStream mstream(m_data, m_dataSize);
7195 bool success = image.LoadFile(mstream, GetImageType());
7196 #else
7197 wxString tempFile;
7198 bool success = wxGetTempFileName(_("image"), tempFile) ;
7199 wxASSERT(success);
7200
7201 if (!WriteBlock(tempFile, m_data, m_dataSize))
7202 {
7203 return false;
7204 }
7205 success = image.LoadFile(tempFile, GetImageType());
7206 wxRemoveFile(tempFile);
7207 #endif
7208
7209 return success;
7210 }
7211
7212 // Write data in hex to a stream
7213 bool wxRichTextImageBlock::WriteHex(wxOutputStream& stream)
7214 {
7215 const int bufSize = 512;
7216 char buf[bufSize+1];
7217
7218 int left = m_dataSize;
7219 int n, i, j;
7220 j = 0;
7221 while (left > 0)
7222 {
7223 if (left*2 > bufSize)
7224 {
7225 n = bufSize; left -= (bufSize/2);
7226 }
7227 else
7228 {
7229 n = left*2; left = 0;
7230 }
7231
7232 char* b = buf;
7233 for (i = 0; i < (n/2); i++)
7234 {
7235 wxDecToHex(m_data[j], b, b+1);
7236 b += 2; j ++;
7237 }
7238
7239 buf[n] = 0;
7240 stream.Write((const char*) buf, n);
7241 }
7242 return true;
7243 }
7244
7245 // Read data in hex from a stream
7246 bool wxRichTextImageBlock::ReadHex(wxInputStream& stream, int length, int imageType)
7247 {
7248 int dataSize = length/2;
7249
7250 if (m_data)
7251 delete[] m_data;
7252
7253 wxChar str[2];
7254 m_data = new unsigned char[dataSize];
7255 int i;
7256 for (i = 0; i < dataSize; i ++)
7257 {
7258 str[0] = (char)stream.GetC();
7259 str[1] = (char)stream.GetC();
7260
7261 m_data[i] = (unsigned char)wxHexToDec(str);
7262 }
7263
7264 m_dataSize = dataSize;
7265 m_imageType = imageType;
7266
7267 return true;
7268 }
7269
7270 // Allocate and read from stream as a block of memory
7271 unsigned char* wxRichTextImageBlock::ReadBlock(wxInputStream& stream, size_t size)
7272 {
7273 unsigned char* block = new unsigned char[size];
7274 if (!block)
7275 return NULL;
7276
7277 stream.Read(block, size);
7278
7279 return block;
7280 }
7281
7282 unsigned char* wxRichTextImageBlock::ReadBlock(const wxString& filename, size_t size)
7283 {
7284 wxFileInputStream stream(filename);
7285 if (!stream.Ok())
7286 return NULL;
7287
7288 return ReadBlock(stream, size);
7289 }
7290
7291 // Write memory block to stream
7292 bool wxRichTextImageBlock::WriteBlock(wxOutputStream& stream, unsigned char* block, size_t size)
7293 {
7294 stream.Write((void*) block, size);
7295 return stream.IsOk();
7296
7297 }
7298
7299 // Write memory block to file
7300 bool wxRichTextImageBlock::WriteBlock(const wxString& filename, unsigned char* block, size_t size)
7301 {
7302 wxFileOutputStream outStream(filename);
7303 if (!outStream.Ok())
7304 return false;
7305
7306 return WriteBlock(outStream, block, size);
7307 }
7308
7309 // Gets the extension for the block's type
7310 wxString wxRichTextImageBlock::GetExtension() const
7311 {
7312 wxImageHandler* handler = wxImage::FindHandler(GetImageType());
7313 if (handler)
7314 return handler->GetExtension();
7315 else
7316 return wxEmptyString;
7317 }
7318
7319 #if wxUSE_DATAOBJ
7320
7321 /*!
7322 * The data object for a wxRichTextBuffer
7323 */
7324
7325 const wxChar *wxRichTextBufferDataObject::ms_richTextBufferFormatId = wxT("wxShape");
7326
7327 wxRichTextBufferDataObject::wxRichTextBufferDataObject(wxRichTextBuffer* richTextBuffer)
7328 {
7329 m_richTextBuffer = richTextBuffer;
7330
7331 // this string should uniquely identify our format, but is otherwise
7332 // arbitrary
7333 m_formatRichTextBuffer.SetId(GetRichTextBufferFormatId());
7334
7335 SetFormat(m_formatRichTextBuffer);
7336 }
7337
7338 wxRichTextBufferDataObject::~wxRichTextBufferDataObject()
7339 {
7340 delete m_richTextBuffer;
7341 }
7342
7343 // after a call to this function, the richTextBuffer is owned by the caller and it
7344 // is responsible for deleting it!
7345 wxRichTextBuffer* wxRichTextBufferDataObject::GetRichTextBuffer()
7346 {
7347 wxRichTextBuffer* richTextBuffer = m_richTextBuffer;
7348 m_richTextBuffer = NULL;
7349
7350 return richTextBuffer;
7351 }
7352
7353 wxDataFormat wxRichTextBufferDataObject::GetPreferredFormat(Direction WXUNUSED(dir)) const
7354 {
7355 return m_formatRichTextBuffer;
7356 }
7357
7358 size_t wxRichTextBufferDataObject::GetDataSize() const
7359 {
7360 if (!m_richTextBuffer)
7361 return 0;
7362
7363 wxString bufXML;
7364
7365 {
7366 wxStringOutputStream stream(& bufXML);
7367 if (!m_richTextBuffer->SaveFile(stream, wxRICHTEXT_TYPE_XML))
7368 {
7369 wxLogError(wxT("Could not write the buffer to an XML stream.\nYou may have forgotten to add the XML file handler."));
7370 return 0;
7371 }
7372 }
7373
7374 #if wxUSE_UNICODE
7375 wxCharBuffer buffer = bufXML.mb_str(wxConvUTF8);
7376 return strlen(buffer) + 1;
7377 #else
7378 return bufXML.Length()+1;
7379 #endif
7380 }
7381
7382 bool wxRichTextBufferDataObject::GetDataHere(void *pBuf) const
7383 {
7384 if (!pBuf || !m_richTextBuffer)
7385 return false;
7386
7387 wxString bufXML;
7388
7389 {
7390 wxStringOutputStream stream(& bufXML);
7391 if (!m_richTextBuffer->SaveFile(stream, wxRICHTEXT_TYPE_XML))
7392 {
7393 wxLogError(wxT("Could not write the buffer to an XML stream.\nYou may have forgotten to add the XML file handler."));
7394 return 0;
7395 }
7396 }
7397
7398 #if wxUSE_UNICODE
7399 wxCharBuffer buffer = bufXML.mb_str(wxConvUTF8);
7400 size_t len = strlen(buffer);
7401 memcpy((char*) pBuf, (const char*) buffer, len);
7402 ((char*) pBuf)[len] = 0;
7403 #else
7404 size_t len = bufXML.Length();
7405 memcpy((char*) pBuf, (const char*) bufXML.c_str(), len);
7406 ((char*) pBuf)[len] = 0;
7407 #endif
7408
7409 return true;
7410 }
7411
7412 bool wxRichTextBufferDataObject::SetData(size_t WXUNUSED(len), const void *buf)
7413 {
7414 delete m_richTextBuffer;
7415 m_richTextBuffer = NULL;
7416
7417 wxString bufXML((const char*) buf, wxConvUTF8);
7418
7419 m_richTextBuffer = new wxRichTextBuffer;
7420
7421 wxStringInputStream stream(bufXML);
7422 if (!m_richTextBuffer->LoadFile(stream, wxRICHTEXT_TYPE_XML))
7423 {
7424 wxLogError(wxT("Could not read the buffer from an XML stream.\nYou may have forgotten to add the XML file handler."));
7425
7426 delete m_richTextBuffer;
7427 m_richTextBuffer = NULL;
7428
7429 return false;
7430 }
7431 return true;
7432 }
7433
7434 #endif
7435 // wxUSE_DATAOBJ
7436
7437
7438 /*
7439 * wxRichTextFontTable
7440 * Manages quick access to a pool of fonts for rendering rich text
7441 */
7442
7443 WX_DECLARE_STRING_HASH_MAP_WITH_DECL(wxFont, wxRichTextFontTableHashMap, class WXDLLIMPEXP_RICHTEXT);
7444
7445 class wxRichTextFontTableData: public wxObjectRefData
7446 {
7447 public:
7448 wxRichTextFontTableData() {}
7449
7450 wxFont FindFont(const wxTextAttr& fontSpec);
7451
7452 wxRichTextFontTableHashMap m_hashMap;
7453 };
7454
7455 wxFont wxRichTextFontTableData::FindFont(const wxTextAttr& fontSpec)
7456 {
7457 wxString facename(fontSpec.GetFontFaceName());
7458 wxString spec(wxString::Format(wxT("%d-%d-%d-%d-%s-%d"), fontSpec.GetFontSize(), fontSpec.GetFontStyle(), fontSpec.GetFontWeight(), (int) fontSpec.GetFontUnderlined(), facename.c_str(), (int) fontSpec.GetFontEncoding()));
7459 wxRichTextFontTableHashMap::iterator entry = m_hashMap.find(spec);
7460
7461 if ( entry == m_hashMap.end() )
7462 {
7463 wxFont font(fontSpec.GetFontSize(), wxDEFAULT, fontSpec.GetFontStyle(), fontSpec.GetFontWeight(), fontSpec.GetFontUnderlined(), facename.c_str());
7464 m_hashMap[spec] = font;
7465 return font;
7466 }
7467 else
7468 {
7469 return entry->second;
7470 }
7471 }
7472
7473 IMPLEMENT_DYNAMIC_CLASS(wxRichTextFontTable, wxObject)
7474
7475 wxRichTextFontTable::wxRichTextFontTable()
7476 {
7477 m_refData = new wxRichTextFontTableData;
7478 }
7479
7480 wxRichTextFontTable::wxRichTextFontTable(const wxRichTextFontTable& table)
7481 {
7482 (*this) = table;
7483 }
7484
7485 wxRichTextFontTable::~wxRichTextFontTable()
7486 {
7487 UnRef();
7488 }
7489
7490 bool wxRichTextFontTable::operator == (const wxRichTextFontTable& table) const
7491 {
7492 return (m_refData == table.m_refData);
7493 }
7494
7495 void wxRichTextFontTable::operator= (const wxRichTextFontTable& table)
7496 {
7497 Ref(table);
7498 }
7499
7500 wxFont wxRichTextFontTable::FindFont(const wxTextAttr& fontSpec)
7501 {
7502 wxRichTextFontTableData* data = (wxRichTextFontTableData*) m_refData;
7503 if (data)
7504 return data->FindFont(fontSpec);
7505 else
7506 return wxFont();
7507 }
7508
7509 void wxRichTextFontTable::Clear()
7510 {
7511 wxRichTextFontTableData* data = (wxRichTextFontTableData*) m_refData;
7512 if (data)
7513 data->m_hashMap.clear();
7514 }
7515
7516 #endif
7517 // wxUSE_RICHTEXT