]>
git.saurik.com Git - wxWidgets.git/blob - src/common/stattextcmn.cpp
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"
42 const wxChar
*wxMarkupEntities
[][wxMARKUP_ENTITY_MAX
] =
44 // the entities handled by SetLabel() when wxST_MARKUP is used and their referenced string
46 { wxT("&"), wxT("<"), wxT(">"), wxT("'"), wxT(""") },
47 { wxT("&"), wxT("<"), wxT(">"), wxT("'"), wxT("\"") }
51 // ----------------------------------------------------------------------------
53 // ----------------------------------------------------------------------------
55 void wxTextWrapper::Wrap(wxWindow
*win
, const wxString
& text
, int widthMax
)
57 const wxChar
*lastSpace
= NULL
;
60 const wxChar
*lineStart
= text
.c_str();
61 for ( const wxChar
*p
= lineStart
; ; p
++ )
63 if ( IsStartOfNewLine() )
72 if ( *p
== _T('\n') || *p
== _T('\0') )
86 if ( widthMax
>= 0 && lastSpace
)
89 win
->GetTextExtent(line
, &width
, NULL
);
91 if ( width
> widthMax
)
93 // remove the last word from this line
94 line
.erase(lastSpace
- lineStart
, p
+ 1 - lineStart
);
97 // go back to the last word of this line which we didn't
102 //else: no wrapping at all or impossible to wrap
108 // ----------------------------------------------------------------------------
109 // wxLabelWrapper: helper class for wxStaticTextBase::Wrap()
110 // ----------------------------------------------------------------------------
112 class wxLabelWrapper
: public wxTextWrapper
115 void WrapLabel(wxWindow
*text
, int widthMax
)
118 Wrap(text
, text
->GetLabel(), widthMax
);
119 text
->SetLabel(m_text
);
123 virtual void OnOutputLine(const wxString
& line
)
128 virtual void OnNewLine()
138 // ----------------------------------------------------------------------------
140 // ----------------------------------------------------------------------------
142 void wxStaticTextBase::Wrap(int width
)
144 wxLabelWrapper wrapper
;
145 wrapper
.WrapLabel(this, width
);
148 wxString
wxStaticTextBase::GetLabelText() const
150 wxString
ret(GetLabel());
152 if (HasFlag(wxST_MARKUP
))
153 ret
= RemoveMarkup(ret
);
154 return RemoveMnemonics(ret
);
158 wxString
wxStaticTextBase::RemoveMarkup(const wxString
& text
)
160 // strip out of "text" the markup for platforms which don't support it natively
161 bool inside_tag
= false;
164 const wxChar
*source
= text
;
165 for (size_t i
=0, max
=text
.length(); i
<max
; i
++)
172 wxLogDebug(wxT("Invalid markup !"));
173 return wxEmptyString
;
181 wxLogDebug(wxT("Invalid markup !"));
182 return wxEmptyString
;
191 wxLogDebug(wxT("Cannot use & as last character of the string '%s'"),
193 return wxEmptyString
;
196 // is this ampersand introducing a mnemonic or rather an entity?
197 bool isMnemonic
= true;
198 for (size_t j
=0; j
< wxMARKUP_ENTITY_MAX
; j
++)
200 const wxChar
*entity
= wxMarkupEntities
[wxMARKUP_ELEMENT_NAME
][j
];
201 size_t entityLen
= wxStrlen(entity
);
203 if (max
- i
>= entityLen
&&
204 wxStrncmp(entity
, &source
[i
], entityLen
) == 0)
206 // replace the &entity; string with the entity reference
207 label
<< wxMarkupEntities
[wxMARKUP_ELEMENT_VALUE
][j
];
209 // little exception: when the entity reference is "&"
210 // (i.e. when entity is "&"), substitute it with &&
211 // instead of a single ampersand:
212 if (*wxMarkupEntities
[wxMARKUP_ELEMENT_VALUE
][j
] == wxT('&'))
214 i
+= entityLen
- 1; // the -1 is because main for()
215 // loop already increments i
237 wxString
wxStaticTextBase::EscapeMarkup(const wxString
& text
)
241 for (const wxChar
*source
= text
; *source
!= wxT('\0'); source
++)
243 bool isEntity
= false;
245 // search in the list of the entities and eventually escape this character
246 for (size_t j
=0; j
< wxMARKUP_ENTITY_MAX
; j
++)
248 if (*source
== *wxMarkupEntities
[wxMARKUP_ELEMENT_VALUE
][j
])
250 ret
<< wxMarkupEntities
[wxMARKUP_ELEMENT_NAME
][j
];
257 ret
<< *source
; // this character does not need to be escaped
265 // ----------------------------------------------------------------------------
266 // wxStaticTextBase - generic implementation for wxST_ELLIPSIZE_* support
267 // ----------------------------------------------------------------------------
269 void wxStaticTextBase::UpdateLabel()
274 wxString newlabel
= GetEllipsizedLabelWithoutMarkup();
276 // we need to touch the "real" label (i.e. the text set inside the control,
277 // using port-specific functions) instead of the string returned by GetLabel().
279 // In fact, we must be careful not to touch the original label passed to
280 // SetLabel() otherwise GetLabel() will behave in a strange way to the user
281 // (e.g. returning a "Ver...ing" instead of "Very long string") !
282 if (newlabel
== DoGetLabel())
284 DoSetLabel(newlabel
);
287 wxString
wxStaticTextBase::GetEllipsizedLabelWithoutMarkup() const
289 // this function should be used only by ports which do not support
290 // ellipsis in static texts: we first remove markup (which cannot
291 // be handled safely by Ellipsize()) and then ellipsize the result.
293 wxString
ret(m_labelOrig
);
295 // the order of the following two blocks is important!
297 if (HasFlag(wxST_MARKUP
))
298 ret
= RemoveMarkup(ret
);
301 ret
= Ellipsize(ret
);
306 #define wxELLIPSE_REPLACEMENT wxT("...")
308 wxString
wxStaticTextBase::Ellipsize(const wxString
& label
) const
310 wxSize
sz(GetSize());
311 if (sz
.GetWidth() < 2 || sz
.GetHeight() < 2)
313 // the size of this window is not valid (yet)
317 wxClientDC
dc(wx_const_cast(wxStaticTextBase
*, this));
318 dc
.SetFont(GetFont());
320 wxArrayInt charOffsets
;
323 // these cannot be cached as they can change because of e.g. a font change
324 int replacementWidth
= dc
.GetTextExtent(wxELLIPSE_REPLACEMENT
).GetWidth();
325 int marginWidth
= dc
.GetCharWidth()*2;
327 // handle correctly labels with newlines
331 for ( const wxChar
*pc
= label
; ; pc
++ )
337 len
= curLine
.length();
339 dc
.GetPartialTextExtents(curLine
, charOffsets
))
341 wxASSERT(charOffsets
.GetCount() == len
);
343 size_t totalWidth
= charOffsets
.Last();
344 if ( totalWidth
> (size_t)sz
.GetWidth() )
346 // we need to ellipsize this row
347 int excessPixels
= totalWidth
- sz
.GetWidth() +
349 marginWidth
; // security margin (NEEDED!)
351 // remove characters in excess
352 size_t initialChar
, // index of first char to erase
353 nChars
; // how many chars do we need to erase?
354 if (HasFlag(wxST_ELLIPSIZE_START
))
358 nChars
< len
&& charOffsets
[nChars
] < excessPixels
;
362 else if (HasFlag(wxST_ELLIPSIZE_MIDDLE
))
364 // the start & end of the removed span of chars
366 size_t endChar
= len
/2;
369 for ( ; removed
< excessPixels
; )
373 // width of the initialChar-th character
374 int width
= charOffsets
[initialChar
] -
375 charOffsets
[initialChar
-1];
377 // remove the initialChar-th character
382 if (endChar
< len
- 1 &&
383 removed
< excessPixels
)
385 // width of the (endChar+1)-th character
386 int width
= charOffsets
[endChar
+1] -
387 charOffsets
[endChar
];
389 // remove the endChar-th character
394 if (initialChar
== 0 && endChar
== len
-1)
402 nChars
= endChar
- initialChar
+ 1;
406 wxASSERT(HasFlag(wxST_ELLIPSIZE_END
));
409 int maxWidth
= totalWidth
- excessPixels
;
412 charOffsets
[initialChar
] < maxWidth
;
416 if (initialChar
== 0)
422 initialChar
--; // go back one character
423 nChars
= len
- initialChar
;
429 // need to remove the entire row!
434 // erase nChars characters after initialChar (included):
435 curLine
.erase(initialChar
, nChars
+1);
437 // if there is space for the replacement dots, add them
438 if (sz
.GetWidth() > replacementWidth
)
439 curLine
.insert(initialChar
, wxELLIPSE_REPLACEMENT
);
442 // if everything was ok, we should have shortened this line
443 // enough to make it fit in sz.GetWidth():
444 wxASSERT(dc
.GetTextExtent(curLine
).GetWidth() < sz
.GetWidth());
448 // add this (ellipsized) row to the rest of the label
449 ret
<< curLine
<< *pc
;
452 if ( *pc
== _T('\0') )
457 // we need to remove mnemonics from the label for
458 // correct calculations
460 // pc+1 is safe: at worst we'll hit the \0
461 if (*(pc
+1) == _T('&'))
462 curLine
+= _T('&'); // && becomes &
463 //else: remove this ampersand
467 // we need also to expand tabs to properly calc their size
469 // Windows natively expands the TABs to 6 spaces. Do the same:
481 #endif // wxUSE_STATTEXT