]>
git.saurik.com Git - wxWidgets.git/blob - user/wxLayout/wxllist.cpp
1 /*-*- c++ -*-********************************************************
2 * wxllist: wxLayoutList, a layout engine for text and graphics *
4 * (C) 1998 by Karsten Ballüder (Ballueder@usa.net) *
7 *******************************************************************/
10 - each Object knows its size and how to draw itself
11 - the list is responsible for calculating positions
12 - the draw coordinates for each object are the top left corner
13 - coordinates only get calculated when things get redrawn
14 - The cursor position is the position before an object, i.e. if the
15 buffer starts with a text-object, cursor 0,0 is just before the
16 first character. For all non-text objects, the cursor positions
17 are 0==before or 1==behind. So that all non-text objects count as
19 - Linebreaks are at the end of a line, that is a line like "abc\n"
20 is four cursor positions long. This makes sure that cursor
21 positions are "as expected", i.e. in "abc\ndef" the 'd' would be
22 at positions (x=0,y=1).
25 The redrawing of the cursor no longer erases it at the last
26 position, because the list gets redrawn anyway.
33 - mouse click positions cursor
34 - selection (SetMark(), GetSelection())
35 - DND acceptance of text / clipboard support
36 - wxlwindow: formatting menu: problem with checked/unchecked consistency gtk bug?
41 #pragma implementation "wxllist.h"
46 # include "gui/wxllist.h"
52 # include "iostream.h"
55 # include <wx/print.h>
59 #define BASELINESTRETCH 12
61 // This should never really get created
62 #define WXLLIST_TEMPFILE "__wxllist.tmp"
65 static const char *g_aTypeStrings
[] =
67 "invalid", "text", "cmd", "icon", "linebreak"
70 # define wxLayoutDebug wxLogDebug
71 # define WXL_VAR(x) cerr << #x " = " << x << endl;
72 # define WXL_DBG_POINT(p) wxLayoutDebug(#p ": (%d, %d)", p.x, p.y)
73 # define WXL_TRACE(f) wxLayoutDebug(#f ": ")
74 # define TypeString(t) g_aTypeStrings[t]
77 wxLayoutObjectBase::Debug(void)
80 wxLayoutDebug("%s: size = %dx%d, pos=%d,%d, bl = %d",
81 TypeString(GetType()), GetSize(&bl
).x
,
83 GetPosition().x
, GetPosition().y
, bl
);
88 # define WXL_DBG_POINT(p)
90 # define ShowCurrentObject()
91 # define TypeString(t) ""
92 inline void wxLayoutDebug(const char *, ...) { }
96 //-------------------------- wxLayoutObjectText
98 wxLayoutObjectText::wxLayoutObjectText(const String
&txt
)
103 m_Position
= wxPoint(-1,-1);
108 wxLayoutObjectText::GetSize(CoordType
*baseLine
) const
110 if(baseLine
) *baseLine
= m_BaseLine
;
111 return wxPoint(m_Width
, m_Height
);
115 wxLayoutObjectText::Draw(wxDC
&dc
, wxPoint
const &translate
)
117 dc
.DrawText(Str(m_Text
), m_Position
.x
+ translate
.x
, m_Position
.y
+translate
.y
);
123 wxLayoutObjectText::Layout(wxDC
&dc
, wxPoint position
, CoordType baseLine
)
127 if(m_Position
.x
!= position
.x
|| m_Position
.y
!= position
.y
)
130 m_Position
= position
;
131 dc
.GetTextExtent(Str(m_Text
),&m_Width
, &m_Height
, &descent
);
132 m_BaseLine
= m_Height
- descent
;
133 if(m_Position
.x
!= position
.x
|| m_Position
.y
!= position
.y
)
137 #ifdef WXLAYOUT_DEBUG
139 wxLayoutObjectText::Debug(void)
141 wxLayoutObjectBase::Debug();
142 wxLayoutDebug(" `%s`", m_Text
.c_str());
146 //-------------------------- wxLayoutObjectIcon
148 wxLayoutObjectIcon::wxLayoutObjectIcon(wxIcon
const &icon
)
150 m_Position
= wxPoint(-1,-1);
151 m_Icon
= new wxIcon(icon
);
154 wxLayoutObjectIcon::wxLayoutObjectIcon(wxIcon
*icon
)
160 wxLayoutObjectIcon::Draw(wxDC
&dc
, wxPoint
const &translate
)
162 dc
.DrawIcon(*m_Icon
,m_Position
.x
+translate
.x
, m_Position
.y
+translate
.y
);
166 wxLayoutObjectIcon::Layout(wxDC
&dc
, wxPoint position
, CoordType baseLine
)
168 if(m_Position
.x
!= position
.x
|| m_Position
.y
!= position
.y
)
170 m_Position
= position
;
174 wxLayoutObjectIcon::GetSize(CoordType
*baseLine
) const
176 if(baseLine
) *baseLine
= m_Icon
->GetHeight();
177 return wxPoint(m_Icon
->GetWidth(), m_Icon
->GetHeight());
180 //-------------------------- wxLayoutObjectCmd
183 wxLayoutObjectCmd::wxLayoutObjectCmd(int size
, int family
, int style
, int
184 weight
, bool underline
,
185 wxColour
const *fg
, wxColour
const *bg
)
188 m_font
= new wxFont(size
,family
,style
,weight
,underline
);
193 wxLayoutObjectCmd::~wxLayoutObjectCmd()
199 wxLayoutObjectCmd::GetStyle(void) const
201 wxLayoutStyleInfo
*si
= new wxLayoutStyleInfo();
204 si
->size
= m_font
->GetPointSize();
205 si
->family
= m_font
->GetFamily();
206 si
->style
= m_font
->GetStyle();
207 si
->underline
= m_font
->GetUnderlined();
208 si
->weight
= m_font
->GetWeight();
210 si
->fg_red
= m_ColourFG
->Red();
211 si
->fg_green
= m_ColourFG
->Green();
212 si
->fg_blue
= m_ColourFG
->Blue();
213 si
->bg_red
= m_ColourBG
->Red();
214 si
->bg_green
= m_ColourBG
->Green();
215 si
->bg_blue
= m_ColourBG
->Blue();
221 wxLayoutObjectCmd::Draw(wxDC
&dc
, wxPoint
const &translate
)
226 dc
.SetTextForeground(*m_ColourFG
);
228 dc
.SetTextBackground(*m_ColourBG
);
231 wxLayoutObjectCmd::Layout(wxDC
&dc
, wxPoint p
, CoordType baseline
)
233 m_Position
= p
; // required so we can find the right object for cursor
234 // this get called, so that recalculation uses right font sizes
235 Draw(dc
,wxPoint(0,0));
238 //-------------------------- wxLayoutList
240 wxLayoutList::wxLayoutList()
242 m_DefaultSetting
= NULL
;
245 m_boldCursor
= FALSE
;
250 wxLayoutList::~wxLayoutList()
253 delete m_DefaultSetting
;
254 // no deletion of objects, they are owned by the list
258 wxLayoutList::LineBreak(void)
260 Insert(new wxLayoutObjectLineBreak
);
264 wxLayoutList::SetFont(int family
, int size
, int style
, int weight
,
265 int underline
, wxColour
const *fg
,
268 if(family
!= -1) m_FontFamily
= family
;
269 if(size
!= -1) m_FontPtSize
= size
;
270 if(style
!= -1) m_FontStyle
= style
;
271 if(weight
!= -1) m_FontWeight
= weight
;
272 if(underline
!= -1) m_FontUnderline
= underline
!= 0;
274 if(fg
!= NULL
) m_ColourFG
= fg
;
275 if(bg
!= NULL
) m_ColourBG
= bg
;
278 new wxLayoutObjectCmd(m_FontPtSize
,m_FontFamily
,m_FontStyle
,m_FontWeight
,m_FontUnderline
,
279 m_ColourFG
, m_ColourBG
));
283 wxLayoutList::SetFont(int family
, int size
, int style
, int weight
,
284 int underline
, char const *fg
, char const *bg
)
292 cfg
= wxTheColourDatabase
->FindColour(fg
);
294 cbg
= wxTheColourDatabase
->FindColour(bg
);
296 SetFont(family
,size
,style
,weight
,underline
,cfg
,cbg
);
300 /// for access by wxLayoutWindow:
302 wxLayoutList::GetSize(CoordType
*max_x
, CoordType
*max_y
,
303 CoordType
*lineHeight
)
306 if(max_x
) *max_x
= m_MaxX
;
307 if(max_y
) *max_y
= m_MaxY
;
308 if(lineHeight
) *lineHeight
= m_LineHeight
;
312 wxLayoutList::ResetSettings(wxDC
&dc
)
314 // setting up the default:
315 dc
.SetTextForeground( *wxBLACK
);
316 dc
.SetTextBackground( *wxWHITE
);
317 dc
.SetBackgroundMode( wxSOLID
); // to enable setting of text background
318 dc
.SetFont( *wxNORMAL_FONT
);
320 m_DefaultSetting
->Draw(dc
,wxPoint(0,0));
324 wxLayoutList::Layout(wxDC
&dc
, wxLayoutMargins
*margins
)
328 // first object in current line
329 wxLayoutObjectList::iterator headOfLine
;
330 // where we draw next
331 wxPoint position
, position_HeadOfLine
;
332 // size of last object
334 CoordType baseLine
= m_FontPtSize
;
335 CoordType baseLineSkip
= (BASELINESTRETCH
* baseLine
)/10;
336 CoordType objBaseLine
= baseLine
;
337 wxLayoutObjectType type
;
339 // we need to count cursor positions
340 wxPoint cursorPos
= wxPoint(0,0);
344 position
.y
= margins
->top
;
345 position
.x
= margins
->left
;
357 position_HeadOfLine
= position
;
364 type
= (*i
)->GetType();
365 (*i
)->Layout(dc
, position
, baseLine
);
366 size
= (*i
)->GetSize(&objBaseLine
);
367 // calculate next object's position:
368 position
.x
+= size
.x
;
370 // do we need to increase the line's height?
371 if(size
.y
> baseLineSkip
)
373 baseLineSkip
= size
.y
;
374 i
= headOfLine
; position
= position_HeadOfLine
;
377 if(objBaseLine
> baseLine
)
379 baseLine
= objBaseLine
;
380 i
= headOfLine
; position
= position_HeadOfLine
;
384 // when we reach here, the coordinates are valid, this part of
385 // the loop gets run only once per object
386 if(position
.x
> m_MaxX
)
388 if(type
== WXLO_TYPE_LINEBREAK
)
390 cursorPos
.x
= 0; cursorPos
.y
++;
393 cursorPos
.x
+= (**i
).CountPositions();
395 // now check whether we have finished handling this line:
396 if(type
== WXLO_TYPE_LINEBREAK
&& i
!= tail())
398 position
.x
= margins
? margins
->left
: 0;
399 position
.y
+= baseLineSkip
;
400 baseLine
= m_FontPtSize
;
401 objBaseLine
= baseLine
; // not all objects set it
402 baseLineSkip
= (BASELINESTRETCH
* baseLine
)/10;
405 position_HeadOfLine
= position
;
407 if(i
== m_CursorObject
)
412 m_MaxY
= position
.y
+ baseLineSkip
;
416 wxLayoutList::Draw(wxDC
&dc
,
417 CoordType fromLine
, CoordType toLine
,
419 wxPoint
const &translate
)
421 //Layout(dc); // FIXME just for now
425 wxLayoutObjectList::iterator i
;
427 if(start
== iterator(NULL
))
429 else // we need to restore font settings
431 for( i
= begin() ; i
!= start
; i
++)
432 if((**i
).GetType() == WXLO_TYPE_CMD
)
433 (**i
).Draw(dc
,translate
); // apply font settings
436 while( start
!= end() && (**start
).GetPosition().y
< fromLine
)
438 if((**start
).GetType() == WXLO_TYPE_CMD
)
439 (**start
).Draw(dc
,translate
); // apply font settings
443 i
!= end() && (toLine
== -1 || (**i
).GetPosition().y
< toLine
) ;
445 (*i
)->Draw(dc
,translate
);
448 /** Erase at least to end of line */
450 wxLayoutList::EraseAndDraw(wxDC
&dc
, iterator start
, wxPoint
const &translate
)
452 //look for begin of line
453 while(start
!= end() && start
!= begin() && (**start
).GetType() !=
456 if(start
== iterator(NULL
))
458 if(start
== iterator(NULL
))
461 wxPoint p
= (**start
).GetPosition();
463 //FIXME: wxGTK: MaxX()/MaxY() broken
464 //WXL_VAR(dc.MaxX()); WXL_VAR(dc.MaxY());
466 dc
.SetBrush(wxBrush(*m_ColourBG
, wxSOLID
));
467 dc
.SetPen(wxPen(*m_ColourBG
,0,wxTRANSPARENT
));
468 dc
.DrawRectangle(p
.x
,p
.y
,2000,2000); //dc.MaxX(),dc.MaxY());
469 Draw(dc
,-1,-1,start
,translate
);
470 //dc.DrawRectangle(p.x,p.y,2000,2000); //dc.MaxX(),dc.MaxY());
475 wxLayoutList::CalculateCursor(wxDC
&dc
)
480 CoordType width
, height
, descent
;
481 CoordType baseLineSkip
= 20; //FIXME
483 int cursorWidth
= m_boldCursor
? 4 : 2;
485 if( m_CursorObject
== iterator(NULL
)) // empty list
487 m_CursorCoords
= wxPoint(0,0);
488 m_CursorSize
= wxPoint(cursorWidth
,baseLineSkip
);
489 m_CursorMoved
= false; // coords are valid
492 wxLayoutObjectBase
&obj
= **m_CursorObject
;
494 m_CursorCoords
= obj
.GetPosition();
495 if(obj
.GetType() == WXLO_TYPE_TEXT
)
497 wxLayoutObjectText
*tobj
= (wxLayoutObjectText
*)&obj
;
498 String
& str
= tobj
->GetText();
499 String sstr
= str
.substr(0,m_CursorOffset
);
500 dc
.GetTextExtent(sstr
,&width
,&height
,&descent
);
501 m_CursorCoords
= wxPoint(m_CursorCoords
.x
+width
,
503 m_CursorSize
= wxPoint(cursorWidth
,height
);
505 else if(obj
.GetType() == WXLO_TYPE_LINEBREAK
)
507 if(m_CursorOffset
== 1) // behind linebreak
508 m_CursorCoords
= wxPoint(0, m_CursorCoords
.y
+ baseLineSkip
);
509 //m_CursorCoords = wxPoint(0, m_CursorCoords.y);
510 m_CursorSize
= wxPoint(cursorWidth
,baseLineSkip
);
514 // this is not necessarily the most "beautiful" solution:
515 //cursorPosition = wxPoint(position.x, position.y);
516 //cursorSize = wxPoint(size.x > 0 ? size.x : 1,size.y > 0 ? size.y : baseLineSkip);
517 m_CursorCoords
= wxPoint(m_CursorCoords
.x
+obj
.GetSize().x
, m_CursorCoords
.y
);
518 m_CursorSize
= wxPoint(cursorWidth
, obj
.GetSize().y
);
519 if(m_CursorSize
.y
< 1) m_CursorSize
.y
= baseLineSkip
;
521 m_CursorMoved
= false; // coords are valid
525 wxLayoutList::DrawCursor(wxDC
&dc
, bool erase
, wxPoint
const &translate
)
533 dc
.Blit(m_CursorCoords
.x
+translate
.x
,
534 m_CursorCoords
.y
+translate
.y
,
535 m_CursorSize
.x
,m_CursorSize
.y
,
541 // erase it at the old position:
542 if(IsDirty() || CursorMoved())
544 // We don't need to erase the cursor because the screen gets
545 // redrawn completely.
546 // DrawCursor(dc,true);
547 // this is needed to update the cursor coordinates
552 wxBitmap
bm(m_CursorSize
.x
+1,m_CursorSize
.y
+1);
553 m_CursorMemDC
.SelectObject(bm
);
554 m_CursorMemDC
.Blit(0, 0,
555 m_CursorSize
.x
, m_CursorSize
.y
,
557 m_CursorCoords
.x
+translate
.x
,
558 m_CursorCoords
.y
+translate
.y
, 0, 0);
561 dc
.SetBrush(*wxBLACK_BRUSH
);
562 dc
.SetPen(wxPen(*wxBLACK
,1,wxSOLID
));
563 dc
.DrawRectangle(m_CursorCoords
.x
+translate
.x
, m_CursorCoords
.y
+translate
.y
,
564 m_CursorSize
.x
, m_CursorSize
.y
);
574 #ifdef WXLAYOUT_DEBUG
576 wxLayoutList::Debug(void)
578 wxLayoutObjectList::iterator i
;
580 wxLayoutDebug("------------------------ debug start ------------------------");
581 for(i
= begin(); i
!= end(); i
++)
583 wxLayoutDebug("-------------------------- list end -------------------------");
585 // show current object:
587 wxLayoutDebug("------------------------- debug end -------------------------");
591 wxLayoutList::ShowCurrentObject()
593 wxLayoutDebug("CursorPos (%d, %d)", (int) m_CursorPos
.x
, (int) m_CursorPos
.y
);
594 wxLayoutDebug("CursorOffset = %d", (int) m_CursorOffset
);
595 wxLayoutDebug("CursorObject = %p", m_CursorObject
);
596 if(m_CursorObject
== iterator(NULL
))
597 wxLayoutDebug("<<no object found>>");
600 if((*m_CursorObject
)->GetType() == WXLO_TYPE_TEXT
)
601 wxLayoutDebug(" \"%s\", offs: %d",
602 ((wxLayoutObjectText
*)(*m_CursorObject
))->GetText().c_str(),
605 wxLayoutDebug(" %s", TypeString((*m_CursorObject
)->GetType()));
607 wxLayoutDebug("Line length: %d", GetLineLength(m_CursorObject
));
613 /******************** editing stuff ********************/
615 // don't change this, I know how to optimise this and will do it real
620 * Finds the object belonging to a given cursor position cpos and
621 * returns an iterator to that object and stores the relative cursor
622 * position in offset.
624 * For linebreaks, the offset can be 0=before or 1=after.
626 * If the cpos coordinates don't exist, they are modified.
629 wxLayoutObjectList::iterator
630 wxLayoutList::FindObjectCursor(wxPoint
*cpos
, CoordType
*offset
)
632 wxPoint object
= wxPoint(0,0); // runs along the objects
634 wxLayoutObjectList::iterator i
, begin_it
;
637 //#ifdef WXLAYOUT_DEBUG
638 // wxLayoutDebug("Looking for object at (%d, %d)", cpos->x, cpos->y);
641 // optimisation: compare to last looked at object:
642 if(cpos
->y
> m_FoundCursor
.y
|| (cpos
->y
== m_FoundCursor
.y
&&
643 cpos
->x
>= m_FoundCursor
.x
))
648 //broken at the moment
649 //begin_it = m_FoundIterator;
650 //m_FoundCursor = *cpos;
653 for(i
= begin_it
; i
!= end() && object
.y
<= cpos
->y
; )
655 width
= (**i
).CountPositions();
656 if(cpos
->y
== object
.y
) // a possible candidate
658 if((**i
).GetType() ==WXLO_TYPE_LINEBREAK
)
660 if(cpos
->x
== object
.x
)
662 if(offset
) *offset
= 0;
663 return m_FoundIterator
= i
;
665 if(offset
) *offset
=1;
667 return m_FoundIterator
= i
;
669 if(cpos
->x
>= object
.x
&& cpos
->x
<= object
.x
+width
) // overlap
671 if(offset
) *offset
= cpos
->x
-object
.x
;
672 //#ifdef WXLAYOUT_DEBUG
673 // wxLayoutDebug(" found object at (%d, %d), type: %s",
674 // object.x, object.y, TypeString((*i)->GetType()));
676 return m_FoundIterator
= i
;
679 // no overlap, increment coordinates
681 if((**i
).GetType() == WXLO_TYPE_LINEBREAK
)
691 //#ifdef WXLAYOUT_DEBUG
692 // wxLayoutDebug(" not found");
694 // return last object, coordinates of that one:
697 return m_FoundIterator
= i
;
698 if((**i
).GetType()==WXLO_TYPE_LINEBREAK
)
702 return m_FoundIterator
= i
;
704 cpos
->x
= object
.x
; // would be the coordinate of next object
706 cpos
->x
+= width
; // last object's width
707 if(*offset
) *offset
= cpos
->x
-object
.x
;
708 return m_FoundIterator
= i
; // not found
712 wxLayoutList::MoveCursor(int dx
, int dy
)
716 m_CursorMoved
= true;
718 enum { up
, down
} direction
;
720 wxPoint newPos
= wxPoint(m_CursorPos
.x
+ dx
,
724 //if(newPos.x < 0) newPos.x = 0;
725 if(newPos
.y
< 0) newPos
.y
= 0;
726 else if(newPos
.y
> m_MaxLine
) newPos
.y
= m_MaxLine
;
728 //FIXME: quick and dirty hack: as last object in buffer should be a
729 // linebreak, we don't allow to go there
730 if(newPos
.y
>= m_MaxLine
)
733 if(newPos
.y
> m_CursorPos
.y
||
734 newPos
.y
== m_CursorPos
.y
&&
735 newPos
.x
>= m_CursorPos
.x
)
740 if ( !m_CursorObject
)
746 // now move cursor forwards until at the new position:
748 // first, go to the right line:
749 while(newPos
.y
!= m_CursorPos
.y
)
751 if(direction
== down
)
754 (**m_CursorObject
).CountPositions() - m_CursorOffset
;
755 if(m_CursorObject
== tail())
756 break; // can't go any further
757 if((**m_CursorObject
).GetType() == WXLO_TYPE_LINEBREAK
758 && m_CursorOffset
== 0)
760 m_CursorPos
.y
++; m_CursorPos
.x
= 0;
762 m_CursorObject
++; m_CursorOffset
= 0;
766 if(m_CursorObject
== begin())
767 break; // can't go any further
769 if((**m_CursorObject
).GetType() == WXLO_TYPE_LINEBREAK
&&
773 m_CursorPos
.x
= GetLineLength(m_CursorObject
);
775 m_CursorPos
.x
-= m_CursorOffset
;
776 m_CursorObject
--; m_CursorOffset
= (**m_CursorObject
).CountPositions();
779 if(newPos
.y
!= m_CursorPos
.y
) // reached begin/end of list,
780 newPos
.y
= m_CursorPos
.y
; // exited by break
782 // now line is right, go to right column:
783 if(dx
== 0) // we are moving up or down only
785 int max_x
= GetLineLength(m_CursorObject
);
786 if(max_x
<= newPos
.x
) // ... so we don't want to cross linebreaks
787 newPos
.x
= max_x
-1; // but just go to the right column
789 direction
= newPos
.x
>= m_CursorPos
.x
? down
: up
;
790 while(newPos
.x
!= m_CursorPos
.x
)
792 if(direction
== down
)
794 diff
= newPos
.x
- m_CursorPos
.x
;
795 if(diff
> (**m_CursorObject
).CountPositions()-m_CursorOffset
)
798 (**m_CursorObject
).CountPositions()-m_CursorOffset
;
799 if((**m_CursorObject
).GetType() == WXLO_TYPE_LINEBREAK
&&
801 m_CursorPos
= wxPoint(0, m_CursorPos
.y
+1);
802 if(m_CursorObject
== tail())
803 break; // cannot go further
804 m_CursorObject
++; m_CursorOffset
= 0;
808 if((**m_CursorObject
).GetType() == WXLO_TYPE_LINEBREAK
)
810 newPos
.y
++; newPos
.x
-= m_CursorPos
.x
+1;
811 m_CursorPos
= wxPoint(0,m_CursorPos
.y
+1);
814 m_CursorPos
.x
+= diff
;
815 m_CursorOffset
+= diff
;
820 if(m_CursorPos
.x
== 0 && m_CursorOffset
== 1 &&
821 (**m_CursorObject
).GetType() == WXLO_TYPE_LINEBREAK
) // can we go further up?
826 m_CursorPos
.x
= GetLineLength(m_CursorObject
)-1;
827 newPos
.x
+= m_CursorPos
.x
+1;
830 diff
= m_CursorPos
.x
- newPos
.x
;
831 if(diff
>= m_CursorOffset
)
833 if(m_CursorObject
== begin())
837 break; // cannot go further
840 m_CursorPos
.x
-= m_CursorOffset
;
841 m_CursorOffset
= (**m_CursorObject
).CountPositions();
845 m_CursorPos
.x
-= diff
;
846 m_CursorOffset
-= diff
;
851 return true; // FIXME: when return what?
854 wxLayoutList::SetCursor(wxPoint
const &p
)
857 m_CursorObject
= FindObjectCursor(&m_CursorPos
, &m_CursorOffset
);
858 m_CursorMoved
= true;
862 wxLayoutList::Delete(CoordType count
)
872 wxLayoutObjectList::iterator i
;
877 startover
: // ugly, but easiest way to do it
879 return; // we cannot delete anything more
881 /* Here we need to treat linebreaks differently.
882 If m_CursorOffset==0 we are before the linebreak, otherwise behind. */
883 if((*i
)->GetType() == WXLO_TYPE_LINEBREAK
)
885 if(m_CursorOffset
== 0)
889 m_CursorObject
= i
; // new i!
890 m_CursorOffset
= 0; // Pos unchanged
892 continue; // we're done
894 else // delete the object behind the linebreak
896 i
++; // we increment and continue as normal
901 else if((*i
)->GetType() == WXLO_TYPE_TEXT
)
903 wxLayoutObjectText
*tobj
= (wxLayoutObjectText
*)*i
;
904 CoordType len
= tobj
->CountPositions();
905 /* If we find the end of a text object, this means that we
906 have to delete from the object following it. */
907 if(len
== m_CursorOffset
)
915 if(m_CursorOffset
== 0 && len
<= count
) // delete this object
924 int todelete
= count
;
925 if(todelete
> len
-m_CursorOffset
)
926 todelete
= len
-m_CursorOffset
;
928 len
= len
- todelete
;
929 tobj
->GetText().erase(m_CursorOffset
,todelete
);
932 return; // we are done
935 else// all other objects: delete the object
936 // this only works as expected if the non-text object has 0/1
937 // as offset values. Not tested with "longer" objects.
939 CoordType len
= (*i
)->CountPositions();
942 count
= count
> len
? count
-= len
: 0;
943 erase(i
); // after this, i is the iterator for the
949 else // delete the following object
951 i
++; // we increment and continue as normal
957 while(count
&& i
!= end());
962 wxLayoutList::Insert(wxLayoutObjectBase
*obj
)
964 wxCHECK_RET( obj
, "no object to insert" );
968 wxLayoutObjectList::iterator i
= m_CursorObject
;
970 if(i
!= iterator(NULL
) && (*obj
).GetType() == WXLO_TYPE_LINEBREAK
)
972 m_CursorPos
.x
= 0; m_CursorPos
.y
++;
978 m_CursorObject
= tail();
980 else if(m_CursorOffset
== 0)
985 // do we have to split a text object?
986 else if((*i
)->GetType() == WXLO_TYPE_TEXT
&& m_CursorOffset
!= (*i
)->CountPositions())
988 wxLayoutObjectText
*tobj
= (wxLayoutObjectText
*) *i
;
989 String left
= tobj
->GetText().substr(0,m_CursorOffset
); // get part before cursor
990 tobj
->GetText() = tobj
->GetText().substr(m_CursorOffset
,(*i
)->CountPositions()-m_CursorOffset
); // keeps the right half
992 m_CursorObject
= i
; // == obj
993 insert(i
,new wxLayoutObjectText(left
)); // inserts before
997 // all other cases, append after object:
998 wxLayoutObjectList::iterator j
= i
; // we want to apend after this object
1008 m_CursorObject
= tail();
1012 if(obj
->GetType() != WXLO_TYPE_LINEBREAK
) // handled separately above
1013 m_CursorPos
.x
+= obj
->CountPositions();
1014 // applies also for linebreak:
1015 m_CursorOffset
= obj
->CountPositions();
1017 if(obj
->GetType() == WXLO_TYPE_LINEBREAK
)
1019 m_CursorMoved
= true;
1023 wxLayoutList::Insert(String
const &text
)
1025 wxLayoutObjectText
*tobj
= NULL
;
1026 wxLayoutObjectList::iterator j
;
1028 // WXL_TRACE(Insert(text));
1035 wxLayoutObjectList::iterator i
= m_CursorObject
;
1039 Insert(new wxLayoutObjectText(text
));
1043 switch((**i
).GetType())
1045 case WXLO_TYPE_TEXT
:
1046 // insert into an existing text object:
1047 tobj
= (wxLayoutObjectText
*)*i
;
1049 tobj
->GetText().insert(m_CursorOffset
,text
);
1051 m_CursorOffset
= m_CursorOffset
+ text
.length();
1052 m_CursorPos
.x
+= text
.length();
1054 case WXLO_TYPE_LINEBREAK
:
1057 if(m_CursorOffset
== 0) // try to append to previous object
1060 if(j
!= end() && (**j
).GetType() == WXLO_TYPE_TEXT
)
1062 tobj
= (wxLayoutObjectText
*)*j
;
1063 tobj
->GetText()+=text
;
1065 m_CursorOffset
= (**j
).CountPositions();
1066 m_CursorPos
.x
+= text
.length();
1070 insert(i
,new wxLayoutObjectText(text
));
1072 m_CursorOffset
= (**i
).CountPositions();
1073 m_CursorPos
.x
+= m_CursorOffset
;
1076 else // offset == 1 : cursor after linebreak
1081 if(j
!= end() && (**j
).GetType() == WXLO_TYPE_TEXT
)
1083 tobj
= (wxLayoutObjectText
*)*j
;
1084 tobj
->GetText()=text
+tobj
->GetText();
1085 m_CursorOffset
= text
.length();
1086 m_CursorPos
.x
+= m_CursorOffset
;
1092 push_back(new wxLayoutObjectText(text
));
1093 m_CursorObject
= tail();
1094 m_CursorOffset
= (**m_CursorObject
).CountPositions();
1095 m_CursorPos
.x
+= text
.length();
1099 insert(j
,new wxLayoutObjectText(text
));
1101 m_CursorOffset
= (**j
).CountPositions();
1102 m_CursorPos
.x
+= text
.length();
1108 m_CursorMoved
= true;
1112 wxLayoutList::GetLineLength(wxLayoutObjectList::iterator i
, CoordType offs
)
1119 if(offs
== 0 && (**i
).GetType() == WXLO_TYPE_LINEBREAK
)
1123 return 0; // at begin of buffer in front of a linebreak
1125 // search backwards for beginning of line:
1126 while(i
!= begin() && (*i
)->GetType() != WXLO_TYPE_LINEBREAK
)
1128 if((*i
)->GetType() == WXLO_TYPE_LINEBREAK
)
1130 // now we can start counting:
1131 while(i
!= end() && (*i
)->GetType() != WXLO_TYPE_LINEBREAK
)
1133 len
+= (*i
)->CountPositions();
1136 len
++; // one extra for the linebreak
1141 wxLayoutList::Clear(int family
, int size
, int style
, int weight
,
1142 int underline
, char const *fg
, char const *bg
)
1145 m_CursorMoved
= true;
1146 m_dirty
= true; // force redraw/recalc
1147 wxLayoutObjectList::iterator i
= begin();
1150 m_CursorMemDC
.SelectObject(bm
);
1152 while(i
!= end()) // == while valid
1156 m_FontPtSize
= size
;
1157 m_FontUnderline
= false;
1158 m_FontFamily
= family
;
1159 m_FontStyle
= style
;
1160 m_FontWeight
= weight
;
1161 m_ColourFG
= wxTheColourDatabase
->FindColour(fg
);
1162 m_ColourBG
= wxTheColourDatabase
->FindColour(bg
);
1164 if(! m_ColourFG
) m_ColourFG
= wxBLACK
;
1165 if(! m_ColourBG
) m_ColourBG
= wxWHITE
;
1167 m_Position
= wxPoint(0,0);
1168 m_CursorPos
= wxPoint(0,0);
1169 m_CursorObject
= iterator(NULL
);
1171 m_CursorSize
= wxPoint(2,(BASELINESTRETCH
*m_FontPtSize
)/10);
1174 m_LineHeight
= (BASELINESTRETCH
*m_FontPtSize
)/10;
1175 m_MaxX
= 0; m_MaxY
= 0;
1178 m_FoundCursor
= wxPoint(0,0);
1179 m_FoundIterator
= begin();
1181 if(m_DefaultSetting
)
1182 delete m_DefaultSetting
;
1184 m_DefaultSetting
= new
1185 wxLayoutObjectCmd(m_FontPtSize
,m_FontFamily
,m_FontStyle
,
1186 m_FontWeight
,m_FontUnderline
,
1187 m_ColourFG
, m_ColourBG
);
1191 wxLayoutObjectBase
*
1192 wxLayoutList::Find(wxPoint coords
) const
1194 wxLayoutObjectList::iterator i
= begin();
1196 wxPoint topleft
, bottomright
;
1198 while(i
!= end()) // == while valid
1200 wxLayoutObjectBase
*object
= *i
;
1201 topleft
= object
->GetPosition();
1202 if(coords
.y
>= topleft
.y
&& coords
.x
>= topleft
.x
)
1204 bottomright
= topleft
;
1205 bottomright
.x
+= object
->GetSize().x
;
1206 bottomright
.y
+= object
->GetSize().y
;
1207 if(coords
.x
<= bottomright
.x
&& coords
.y
<= bottomright
.y
)
1217 wxLayoutList::SetWrapMargin(long n
)
1223 wxLayoutList::WrapLine(void)
1225 wxASSERT(m_CursorObject
);
1227 iterator i
= m_CursorObject
;
1229 if(!DoWordWrap() || !i
) // empty list
1231 int cursorpos
= m_CursorPos
.x
, cpos
, offset
;
1233 if(cursorpos
< m_WrapMargin
)
1238 // find the right object to break:
1239 // is it the current one?
1242 cpos
= cursorpos
-m_CursorOffset
;
1243 while(i
!= begin() && cpos
>= m_WrapMargin
)
1246 cpos
-= (**i
).CountPositions();
1248 // now i is the object to break and cpos its position
1250 offset
= m_WrapMargin
- cpos
;
1251 wxASSERT(offset
<= (**i
).CountPositions());
1254 if((**i
).GetType() == WXLO_TYPE_TEXT
)
1256 wxLayoutObjectText
&t
= *(wxLayoutObjectText
*)*i
;
1257 for(; offset
> 0; offset
--)
1258 if(t
.GetText().c_str()[offset
] == ' ' || t
.GetText().c_str()[offset
] == '\t')
1260 String left
= t
.GetText().substr(0,offset
); // get part before cursor
1261 t
.GetText() = t
.GetText().substr(offset
+1,t
.CountPositions()-offset
-1); // keeps the right halve
1262 insert(i
,new wxLayoutObjectLineBreak
);
1263 insert(i
,new wxLayoutObjectText(left
)); // inserts before
1268 // only insert a line break if there isn't already one
1269 iterator j
= i
; j
--;
1270 if(j
&& j
!= begin() && (**j
).GetType() != WXLO_TYPE_LINEBREAK
)
1271 insert(i
,new wxLayoutObjectLineBreak
);
1273 return; // do nothing
1277 insert(i
,new wxLayoutObjectLineBreak
);
1280 m_CursorPos
.x
-= offset
;
1281 m_CursorOffset
-= offset
;
1283 /******************** printing stuff ********************/
1285 wxLayoutPrintout::wxLayoutPrintout(wxLayoutList
&llist
,
1286 wxString
const & title
)
1293 bool wxLayoutPrintout::OnPrintPage(int page
)
1298 DrawHeader(*dc
,wxPoint(m_Margins
.left
,m_Margins
.top
/2),wxPoint(m_Margins
.right
,m_Margins
.top
),page
);
1300 top
= (page
- 1)*m_PrintoutHeight
;
1301 bottom
= top
+ m_PrintoutHeight
;
1302 // SetDeviceOrigin() doesn't work here, so we need to manually
1303 // translate all coordinates.
1304 wxPoint
translate(m_Margins
.left
,-top
+m_Margins
.top
);
1305 m_llist
->Draw(*dc
,top
,bottom
,wxLayoutObjectList::iterator(NULL
),translate
);
1312 bool wxLayoutPrintout::OnBeginDocument(int startPage
, int endPage
)
1314 if (!wxPrintout::OnBeginDocument(startPage
, endPage
))
1321 wxLayoutPrintout::OnPreparePrinting(void)
1327 void wxLayoutPrintout::GetPageInfo(int *minPage
, int *maxPage
, int *selPageFrom
, int *selPageTo
)
1329 // ugly hack to get number of pages
1331 wxPrinterDC
psdc("","",WXLLIST_TEMPFILE
,false);
1333 wxPostScriptDC
psdc(WXLLIST_TEMPFILE
,false);
1335 psdc
.GetSize(&m_PageWidth
, &m_PageHeight
); // that's all we need it for
1337 // We do 5% margins on top and bottom, and a 5% high header line.
1338 m_Margins
.top
= m_PageHeight
/ 10 ; // 10%, half of it header
1339 m_Margins
.bottom
= m_PageHeight
- m_PageHeight
/ 20; // 95%
1340 // On the sides we reserve 10% each for the margins.
1341 m_Margins
.left
= m_PageWidth
/ 10;
1342 m_Margins
.right
= m_PageWidth
- m_PageWidth
/ 10;
1344 // This is the length of the printable area.
1345 m_PrintoutHeight
= m_PageHeight
- (int) (m_PageHeight
* 0.15);
1347 //FIXME this is wrong but not used at the moment
1348 m_PageWidth
= m_Margins
.right
- m_Margins
.left
;
1350 m_NumOfPages
= (int)( m_llist
->GetSize().y
/ (float)(m_PrintoutHeight
) + 0.5);
1352 *maxPage
= m_NumOfPages
;
1355 *selPageTo
= m_NumOfPages
;
1356 wxRemoveFile(WXLLIST_TEMPFILE
);
1359 bool wxLayoutPrintout::HasPage(int pageNum
)
1361 return pageNum
<= m_NumOfPages
;
1366 wxLayoutPrintout::DrawHeader(wxDC
&dc
,
1367 wxPoint topleft
, wxPoint bottomright
,
1370 // make backups of all essential parameters
1371 const wxBrush
& brush
= dc
.GetBrush();
1372 const wxPen
& pen
= dc
.GetPen();
1373 const wxFont
& font
= dc
.GetFont();
1375 dc
.SetBrush(*wxWHITE_BRUSH
);
1376 dc
.SetPen(wxPen(*wxBLACK
,0,wxSOLID
));
1377 dc
.DrawRoundedRectangle(topleft
.x
,
1378 topleft
.y
,bottomright
.x
-topleft
.x
,
1379 bottomright
.y
-topleft
.y
);
1380 dc
.SetBrush(*wxBLACK_BRUSH
);
1381 wxFont myfont
= wxFont((WXLO_DEFAULTFONTSIZE
*12)/10,
1382 wxSWISS
,wxNORMAL
,wxBOLD
,false,"Helvetica");
1386 page
= "9999/9999 "; // many pages...
1388 dc
.GetTextExtent(page
,&w
,&h
);
1389 page
.Printf("%d/%d", pageno
, (int) m_NumOfPages
);
1390 dc
.DrawText(page
,bottomright
.x
-w
,topleft
.y
+h
/2);
1391 dc
.GetTextExtent("XXXX", &w
,&h
);
1392 dc
.DrawText(m_title
, topleft
.x
+w
,topleft
.y
+h
/2);
1402 wxLayoutList::MakePrintout(wxString
const &name
)
1404 return new wxLayoutPrintout(*this,name
);