]>
git.saurik.com Git - wxWidgets.git/blob - src/common/stattextcmn.cpp
33992dc2463958f696124e3eadb0f70789e1b14c
1 /////////////////////////////////////////////////////////////////////////////
2 // Name: src/common/stattextcmn.cpp
3 // Purpose: common (to all ports) wxStaticText functions
4 // Author: Vadim Zeitlin, Francesco Montorsi
5 // Created: 2007-01-07 (extracted from dlgcmn.cpp)
7 // Copyright: (c) 1999-2006 Vadim Zeitlin
8 // (c) 2007 Francesco Montorsi
9 // Licence: wxWindows licence
10 /////////////////////////////////////////////////////////////////////////////
12 // ============================================================================
14 // ============================================================================
16 // ----------------------------------------------------------------------------
18 // ----------------------------------------------------------------------------
20 // For compilers that support precompilation, includes "wx.h".
21 #include "wx/wxprec.h"
27 #include "wx/private/stattext.h"
30 #include "wx/button.h"
31 #include "wx/dcclient.h"
34 #include "wx/settings.h"
35 #include "wx/stattext.h"
37 #include "wx/containr.h"
40 const wxChar
*wxMarkupEntities
[][wxMARKUP_ENTITY_MAX
] =
42 // the entities handled by SetLabel() when wxST_MARKUP is used and their referenced string
44 { wxT("&"), wxT("<"), wxT(">"), wxT("'"), wxT(""") },
45 { wxT("&"), wxT("<"), wxT(">"), wxT("'"), wxT("\"") }
50 // ----------------------------------------------------------------------------
52 // ----------------------------------------------------------------------------
54 void wxTextWrapper::Wrap(wxWindow
*win
, const wxString
& text
, int widthMax
)
58 wxString::const_iterator lastSpace
= text
.end();
59 wxString::const_iterator lineStart
= text
.begin();
60 for ( wxString::const_iterator p
= lineStart
; ; ++p
)
62 if ( IsStartOfNewLine() )
66 lastSpace
= text
.end();
71 if ( p
== text
.end() || *p
== _T('\n') )
75 if ( p
== text
.end() )
85 if ( widthMax
>= 0 && lastSpace
!= text
.end() )
88 win
->GetTextExtent(line
, &width
, NULL
);
90 if ( width
> widthMax
)
92 // remove the last word from this line
93 line
.erase(lastSpace
- lineStart
, p
+ 1 - lineStart
);
96 // go back to the last word of this line which we didn't
101 //else: no wrapping at all or impossible to wrap
107 // ----------------------------------------------------------------------------
108 // wxLabelWrapper: helper class for wxStaticTextBase::Wrap()
109 // ----------------------------------------------------------------------------
111 class wxLabelWrapper
: public wxTextWrapper
114 void WrapLabel(wxWindow
*text
, int widthMax
)
117 Wrap(text
, text
->GetLabel(), widthMax
);
118 text
->SetLabel(m_text
);
122 virtual void OnOutputLine(const wxString
& line
)
127 virtual void OnNewLine()
137 // ----------------------------------------------------------------------------
139 // ----------------------------------------------------------------------------
141 void wxStaticTextBase::Wrap(int width
)
143 wxLabelWrapper wrapper
;
144 wrapper
.WrapLabel(this, width
);
147 wxString
wxStaticTextBase::GetLabelText() const
149 wxString
ret(GetLabel());
151 if (HasFlag(wxST_MARKUP
))
152 ret
= RemoveMarkup(ret
);
153 return RemoveMnemonics(ret
);
157 wxString
wxStaticTextBase::GetLabelText(const wxString
& label
)
160 wxString ret
= RemoveMarkup(label
);
161 return RemoveMnemonics(ret
);
165 wxString
wxStaticTextBase::RemoveMarkup(const wxString
& text
)
167 // strip out of "text" the markup for platforms which don't support it natively
168 bool inside_tag
= false;
171 for ( wxString::const_iterator source
= text
.begin();
172 source
!= text
.end(); ++source
)
174 switch ( (*source
).GetValue() )
179 wxLogDebug(wxT("Invalid markup !"));
180 return wxEmptyString
;
188 wxLogDebug(wxT("Invalid markup !"));
189 return wxEmptyString
;
196 if ( source
+1 == text
.end() )
198 wxLogDebug(wxT("Cannot use & as last character of the string '%s'"),
200 return wxEmptyString
;
203 // is this ampersand introducing a mnemonic or rather an entity?
204 bool isMnemonic
= true;
205 size_t distanceFromEnd
= text
.end() - source
;
206 for (size_t j
=0; j
< wxMARKUP_ENTITY_MAX
; j
++)
208 const wxChar
*entity
= wxMarkupEntities
[wxMARKUP_ELEMENT_NAME
][j
];
209 size_t entityLen
= wxStrlen(entity
);
211 if (distanceFromEnd
>= entityLen
&&
212 wxString(source
, source
+ entityLen
) == entity
)
214 // replace the &entity; string with the entity reference
215 label
<< wxMarkupEntities
[wxMARKUP_ELEMENT_VALUE
][j
];
216 // little exception: when the entity reference is
217 // "&" (i.e. when entity is "&"), substitute it
218 // with && instead of a single ampersand:
219 if (*wxMarkupEntities
[wxMARKUP_ELEMENT_VALUE
][j
] == wxT('&'))
221 // the -1 is because main for() loop already
223 source
+= entityLen
- 1;
245 wxString
wxStaticTextBase::EscapeMarkup(const wxString
& text
)
249 for (wxString::const_iterator source
= text
.begin();
250 source
!= text
.end(); ++source
)
252 bool isEntity
= false;
254 // search in the list of the entities and eventually escape this character
255 for (size_t j
=0; j
< wxMARKUP_ENTITY_MAX
; j
++)
257 if (*source
== *wxMarkupEntities
[wxMARKUP_ELEMENT_VALUE
][j
])
259 ret
<< wxMarkupEntities
[wxMARKUP_ELEMENT_NAME
][j
];
266 ret
<< *source
; // this character does not need to be escaped
272 #define wxELLIPSE_REPLACEMENT wxT("...")
275 wxString
wxStaticTextBase::Ellipsize(const wxString
& label
, const wxDC
& dc
,
276 wxEllipsizeMode mode
, int maxFinalWidth
)
278 wxArrayInt charOffsets
;
281 // these cannot be cached as they can change because of e.g. a font change
282 int replacementWidth
= dc
.GetTextExtent(wxELLIPSE_REPLACEMENT
).GetWidth();
283 int marginWidth
= dc
.GetCharWidth()*2;
285 // NB: we must handle correctly labels with newlines:
289 for ( wxString::const_iterator pc
= label
.begin(); ; ++pc
)
291 if ( pc
== label
.end() || *pc
== _T('\n') )
293 len
= curLine
.length();
295 dc
.GetPartialTextExtents(curLine
, charOffsets
))
297 wxASSERT(charOffsets
.GetCount() == len
);
299 size_t totalWidth
= charOffsets
.Last();
300 if ( totalWidth
> (size_t)maxFinalWidth
)
302 // we need to ellipsize this row
303 int excessPixels
= totalWidth
- maxFinalWidth
+
305 marginWidth
; // security margin (NEEDED!)
307 // remove characters in excess
308 size_t initialChar
, // index of first char to erase
309 nChars
; // how many chars do we need to erase?
310 if (mode
== wxST_ELLIPSIZE_START
)
314 nChars
< len
&& charOffsets
[nChars
] < excessPixels
;
318 else if (mode
== wxST_ELLIPSIZE_MIDDLE
)
320 // the start & end of the removed span of chars
322 size_t endChar
= len
/2;
325 for ( ; removed
< excessPixels
; )
329 // width of the initialChar-th character
330 int width
= charOffsets
[initialChar
] -
331 charOffsets
[initialChar
-1];
333 // remove the initialChar-th character
338 if (endChar
< len
- 1 &&
339 removed
< excessPixels
)
341 // width of the (endChar+1)-th character
342 int width
= charOffsets
[endChar
+1] -
343 charOffsets
[endChar
];
345 // remove the endChar-th character
350 if (initialChar
== 0 && endChar
== len
-1)
358 nChars
= endChar
- initialChar
+ 1;
362 wxASSERT(mode
== wxST_ELLIPSIZE_END
);
365 int maxWidth
= totalWidth
- excessPixels
;
368 charOffsets
[initialChar
] < maxWidth
;
372 if (initialChar
== 0)
378 initialChar
--; // go back one character
379 nChars
= len
- initialChar
;
385 // need to remove the entire row!
390 // erase nChars characters after initialChar (included):
391 curLine
.erase(initialChar
, nChars
+1);
393 // if there is space for the replacement dots, add them
394 if (maxFinalWidth
> replacementWidth
)
395 curLine
.insert(initialChar
, wxELLIPSE_REPLACEMENT
);
398 // if everything was ok, we should have shortened this line
399 // enough to make it fit in sz.maxFinalWidth:
400 wxASSERT(dc
.GetTextExtent(curLine
).GetWidth() < maxFinalWidth
);
404 // add this (ellipsized) row to the rest of the label
406 if ( pc
== label
.end() )
416 // we need to remove mnemonics from the label for correct calculations
417 else if ( *pc
== _T('&') )
419 // pc+1 is safe: at worst we'll be at end()
420 wxString::const_iterator next
= pc
+ 1;
421 if ( next
!= label
.end() && *next
== _T('&') )
422 curLine
+= _T('&'); // && becomes &
423 //else: remove this ampersand
425 // we need also to expand tabs to properly calc their size
426 else if ( *pc
== _T('\t') )
428 // Windows natively expands the TABs to 6 spaces. Do the same:
442 // ----------------------------------------------------------------------------
443 // wxStaticTextBase - generic implementation for wxST_ELLIPSIZE_* support
444 // ----------------------------------------------------------------------------
446 void wxStaticTextBase::UpdateLabel()
451 wxString newlabel
= GetEllipsizedLabelWithoutMarkup();
453 // we need to touch the "real" label (i.e. the text set inside the control,
454 // using port-specific functions) instead of the string returned by GetLabel().
456 // In fact, we must be careful not to touch the original label passed to
457 // SetLabel() otherwise GetLabel() will behave in a strange way to the user
458 // (e.g. returning a "Ver...ing" instead of "Very long string") !
459 if (newlabel
== DoGetLabel())
461 DoSetLabel(newlabel
);
464 wxString
wxStaticTextBase::GetEllipsizedLabelWithoutMarkup() const
466 // this function should be used only by ports which do not support
467 // ellipsis in static texts: we first remove markup (which cannot
468 // be handled safely by Ellipsize()) and then ellipsize the result.
470 wxString
ret(m_labelOrig
);
472 // the order of the following two blocks is important!
474 if (HasFlag(wxST_MARKUP
))
475 ret
= RemoveMarkup(ret
);
478 ret
= Ellipsize(ret
);
483 wxString
wxStaticTextBase::Ellipsize(const wxString
& label
) const
485 wxSize
sz(GetSize());
486 if (sz
.GetWidth() < 2 || sz
.GetHeight() < 2)
488 // the size of this window is not valid (yet)
492 wxClientDC
dc(const_cast<wxStaticTextBase
*>(this));
493 dc
.SetFont(GetFont());
495 wxEllipsizeMode mode
;
496 if (HasFlag(wxST_ELLIPSIZE_START
)) mode
= wxST_ELLIPSIZE_START
;
497 else if (HasFlag(wxST_ELLIPSIZE_MIDDLE
)) mode
= wxST_ELLIPSIZE_MIDDLE
;
498 else if (HasFlag(wxST_ELLIPSIZE_END
)) mode
= wxST_ELLIPSIZE_END
;
500 return Ellipsize(label
, dc
, mode
, sz
.GetWidth());
503 #endif // wxUSE_STATTEXT