]>
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