1 ///////////////////////////////////////////////////////////////////////////////
2 // Name: src/common/textmeasurecmn.cpp
3 // Purpose: wxTextMeasureBase implementation
4 // Author: Manuel Martin
7 // Copyright: (c) 1997-2012 wxWidgets team
8 // Licence: wxWindows licence
9 ///////////////////////////////////////////////////////////////////////////////
11 // ============================================================================
13 // ============================================================================
15 // ----------------------------------------------------------------------------
17 // ----------------------------------------------------------------------------
19 // for compilers that support precompilation, includes "wx.h".
20 #include "wx/wxprec.h"
24 #include "wx/window.h"
27 #include "wx/private/textmeasure.h"
29 // ============================================================================
30 // wxTextMeasureBase implementation
31 // ============================================================================
33 wxTextMeasureBase::wxTextMeasureBase(const wxDC
*dc
, const wxFont
*theFont
)
38 wxASSERT_MSG( dc
, wxS("wxTextMeasure needs a valid wxDC") );
40 // By default, use wxDC version, we'll explicitly reset this to false in
41 // the derived classes if the DC is of native variety.
45 wxTextMeasureBase::wxTextMeasureBase(const wxWindow
*win
, const wxFont
*theFont
)
50 wxASSERT_MSG( win
, wxS("wxTextMeasure needs a valid wxWindow") );
52 // We don't have any wxDC so we can't forward to it.
56 wxFont
wxTextMeasureBase::GetFont() const
58 return m_font
? *m_font
59 : m_win
? m_win
->GetFont()
63 void wxTextMeasureBase::CallGetTextExtent(const wxString
& string
,
67 wxCoord
*externalLeading
)
70 m_dc
->GetTextExtent(string
, width
, height
, descent
, externalLeading
);
72 DoGetTextExtent(string
, width
, height
, descent
, externalLeading
);
75 void wxTextMeasureBase::GetTextExtent(const wxString
& string
,
79 wxCoord
*externalLeading
)
81 // To make the code simpler, make sure that the width and height pointers
82 // are always valid, even if they point to dummy variables.
83 int unusedWidth
, unusedHeight
;
87 height
= &unusedHeight
;
97 MeasuringGuard
guard(*this);
99 CallGetTextExtent(string
, width
, height
, descent
, externalLeading
);
102 void wxTextMeasureBase::GetMultiLineTextExtent(const wxString
& text
,
105 wxCoord
*heightOneLine
)
107 MeasuringGuard
guard(*this);
109 wxCoord widthTextMax
= 0, widthLine
,
110 heightTextTotal
= 0, heightLineDefault
= 0, heightLine
= 0;
113 for ( wxString::const_iterator pc
= text
.begin(); ; ++pc
)
115 if ( pc
== text
.end() || *pc
== wxS('\n') )
117 if ( curLine
.empty() )
119 // we can't use GetTextExtent - it will return 0 for both width
120 // and height and an empty line should count in height
123 // assume that this line has the same height as the previous
125 if ( !heightLineDefault
)
126 heightLineDefault
= heightLine
;
128 if ( !heightLineDefault
)
130 // but we don't know it yet - choose something reasonable
132 CallGetTextExtent(wxS("W"), &dummy
, &heightLineDefault
);
135 heightTextTotal
+= heightLineDefault
;
139 CallGetTextExtent(curLine
, &widthLine
, &heightLine
);
140 if ( widthLine
> widthTextMax
)
141 widthTextMax
= widthLine
;
142 heightTextTotal
+= heightLine
;
145 if ( pc
== text
.end() )
161 *width
= widthTextMax
;
163 *height
= heightTextTotal
;
165 *heightOneLine
= heightLine
;
168 wxSize
wxTextMeasureBase::GetLargestStringExtent(size_t n
,
169 const wxString
* strings
)
171 MeasuringGuard
guard(*this);
173 wxCoord w
, h
, widthMax
= 0, heightMax
= 0;
174 for ( size_t i
= 0; i
< n
; ++i
)
176 CallGetTextExtent(strings
[i
], &w
, &h
);
184 return wxSize(widthMax
, heightMax
);
187 bool wxTextMeasureBase::GetPartialTextExtents(const wxString
& text
,
195 MeasuringGuard
guard(*this);
197 widths
.Add(0, text
.length());
199 return DoGetPartialTextExtents(text
, widths
, scaleX
);
202 // ----------------------------------------------------------------------------
203 // Generic and inefficient DoGetPartialTextExtents() implementation.
204 // ----------------------------------------------------------------------------
206 // Each element of the widths array will be the width of the string up to and
207 // including the corresponding character in text. This is the generic
208 // implementation, the port-specific classes should do this with native APIs
209 // if available and if faster. Note: pango_layout_index_to_pos is much slower
210 // than calling GetTextExtent!!
217 FontWidthCache() : m_scaleX(1), m_widths(NULL
) { }
218 ~FontWidthCache() { delete []m_widths
; }
223 m_widths
= new int[FWC_SIZE
];
225 memset(m_widths
, 0, sizeof(int)*FWC_SIZE
);
233 static FontWidthCache s_fontWidthCache
;
235 bool wxTextMeasureBase::DoGetPartialTextExtents(const wxString
& text
,
241 // reset the cache if font or horizontal scale have changed
242 if ( !s_fontWidthCache
.m_widths
||
243 !wxIsSameDouble(s_fontWidthCache
.m_scaleX
, scaleX
) ||
244 (s_fontWidthCache
.m_font
!= *m_font
) )
246 s_fontWidthCache
.Reset();
247 s_fontWidthCache
.m_font
= *m_font
;
248 s_fontWidthCache
.m_scaleX
= scaleX
;
251 // Calculate the position of each character based on the widths of
252 // the previous characters. This is inexact for not fixed fonts.
254 for ( wxString::const_iterator it
= text
.begin();
258 const wxChar c
= *it
;
259 unsigned int c_int
= (unsigned int)c
;
262 if ((c_int
< FWC_SIZE
) && (s_fontWidthCache
.m_widths
[c_int
] != 0))
264 w
= s_fontWidthCache
.m_widths
[c_int
];
268 DoGetTextExtent(c
, &w
, NULL
);
269 if (c_int
< FWC_SIZE
)
270 s_fontWidthCache
.m_widths
[c_int
] = w
;
274 widths
[n
++] = totalWidth
;