]> git.saurik.com Git - wxWidgets.git/blob - src/generic/htmllbox.cpp
Added support for delayed deactivation of windows (for MDI)
[wxWidgets.git] / src / generic / htmllbox.cpp
1 ///////////////////////////////////////////////////////////////////////////////
2 // Name: generic/htmllbox.cpp
3 // Purpose: implementation of wxHtmlListBox
4 // Author: Vadim Zeitlin
5 // Modified by:
6 // Created: 31.05.03
7 // RCS-ID: $Id$
8 // Copyright: (c) 2003 Vadim Zeitlin <vadim@wxwindows.org>
9 // License: wxWindows license
10 ///////////////////////////////////////////////////////////////////////////////
11
12 // ============================================================================
13 // declarations
14 // ============================================================================
15
16 // ----------------------------------------------------------------------------
17 // headers
18 // ----------------------------------------------------------------------------
19
20 // For compilers that support precompilation, includes "wx.h".
21 #include "wx/wxprec.h"
22
23 #ifdef __BORLANDC__
24 #pragma hdrstop
25 #endif
26
27 #ifndef WX_PRECOMP
28 #include "wx/dcclient.h"
29 #endif //WX_PRECOMP
30
31 #include "wx/htmllbox.h"
32
33 #include "wx/html/htmlcell.h"
34 #include "wx/html/winpars.h"
35
36 // this hack forces the linker to always link in m_* files
37 #include "wx/html/forcelnk.h"
38 FORCE_WXHTML_MODULES()
39
40 // ----------------------------------------------------------------------------
41 // private classes
42 // ----------------------------------------------------------------------------
43
44 // this class is used by wxHtmlListBox to cache the parsed representation of
45 // the items to avoid doing it anew each time an item must be drawn
46 class wxHtmlListBoxCache
47 {
48 public:
49 wxHtmlListBoxCache()
50 {
51 for ( size_t n = 0; n < SIZE; n++ )
52 {
53 m_items[n] = (size_t)-1;
54 m_cells[n] = NULL;
55 }
56
57 m_next = 0;
58 }
59
60 ~wxHtmlListBoxCache()
61 {
62 for ( size_t n = 0; n < SIZE; n++ )
63 {
64 delete m_cells[n];
65 }
66 }
67
68 // completely invalidate the cache
69 void Clear()
70 {
71 for ( size_t n = 0; n < SIZE; n++ )
72 {
73 m_items[n] = (size_t)-1;
74 delete m_cells[n];
75 m_cells[n] = NULL;
76 }
77 }
78
79 // return the cached cell for this index or NULL if none
80 wxHtmlCell *Get(size_t item) const
81 {
82 for ( size_t n = 0; n < SIZE; n++ )
83 {
84 if ( m_items[n] == item )
85 return m_cells[n];
86 }
87
88 return NULL;
89 }
90
91 // returns true if we already have this item cached
92 bool Has(size_t item) const { return Get(item) != NULL; }
93
94 // ensure that the item is cached
95 void Store(size_t item, wxHtmlCell *cell)
96 {
97 delete m_cells[m_next];
98 m_cells[m_next] = cell;
99 m_items[m_next] = item;
100
101 // advance to the next item wrapping around if there are no more
102 if ( ++m_next == SIZE )
103 m_next = 0;
104 }
105
106 private:
107 // the max number of the items we cache
108 enum { SIZE = 50 };
109
110 // the index of the LRU (oldest) cell
111 size_t m_next;
112
113 // the parsed representation of the cached item or NULL
114 wxHtmlCell *m_cells[SIZE];
115
116 // the index of the currently cached item (only valid if m_cells != NULL)
117 size_t m_items[SIZE];
118 };
119
120 // ----------------------------------------------------------------------------
121 // event tables
122 // ----------------------------------------------------------------------------
123
124 BEGIN_EVENT_TABLE(wxHtmlListBox, wxVListBox)
125 EVT_SIZE(wxHtmlListBox::OnSize)
126 END_EVENT_TABLE()
127
128 // ============================================================================
129 // implementation
130 // ============================================================================
131
132 // ----------------------------------------------------------------------------
133 // wxHtmlListBox creation
134 // ----------------------------------------------------------------------------
135
136 void wxHtmlListBox::Init()
137 {
138 m_htmlParser = NULL;
139 m_cache = new wxHtmlListBoxCache;
140 }
141
142 bool wxHtmlListBox::Create(wxWindow *parent,
143 wxWindowID id,
144 const wxPoint& pos,
145 const wxSize& size,
146 long style,
147 const wxString& name)
148 {
149 return wxVListBox::Create(parent, id, pos, size, style, name);
150 }
151
152 wxHtmlListBox::~wxHtmlListBox()
153 {
154 delete m_cache;
155 if ( m_htmlParser )
156 {
157 delete m_htmlParser->GetDC();
158 delete m_htmlParser;
159 }
160 }
161
162 // ----------------------------------------------------------------------------
163 // wxHtmlListBox items markup
164 // ----------------------------------------------------------------------------
165
166 wxString wxHtmlListBox::OnGetItemMarkup(size_t n) const
167 {
168 // we don't even need to wrap the value returned by OnGetItem() inside
169 // "<html><body>" and "</body></html>" because wxHTML can parse it even
170 // without these tags
171 return OnGetItem(n);
172 }
173
174 // ----------------------------------------------------------------------------
175 // wxHtmlListBox cache handling
176 // ----------------------------------------------------------------------------
177
178 void wxHtmlListBox::CacheItem(size_t n) const
179 {
180 if ( !m_cache->Has(n) )
181 {
182 if ( !m_htmlParser )
183 {
184 wxHtmlListBox *self = wxConstCast(this, wxHtmlListBox);
185
186 self->m_htmlParser = new wxHtmlWinParser;
187 m_htmlParser->SetDC(new wxClientDC(self));
188 }
189
190 wxHtmlContainerCell *cell = (wxHtmlContainerCell *)m_htmlParser->
191 Parse(OnGetItemMarkup(n));
192 wxCHECK_RET( cell, _T("wxHtmlParser::Parse() returned NULL?") );
193
194 cell->Layout(GetClientSize().x - 2*GetMargins().x);
195
196 m_cache->Store(n, cell);
197 }
198 }
199
200 void wxHtmlListBox::OnSize(wxSizeEvent& event)
201 {
202 // we need to relayout all the cached cells
203 m_cache->Clear();
204
205 event.Skip();
206 }
207
208 void wxHtmlListBox::RefreshAll()
209 {
210 m_cache->Clear();
211
212 wxVListBox::RefreshAll();
213 }
214
215 // ----------------------------------------------------------------------------
216 // wxHtmlListBox implementation of wxVListBox pure virtuals
217 // ----------------------------------------------------------------------------
218
219 void wxHtmlListBox::OnDrawItem(wxDC& dc, const wxRect& rect, size_t n) const
220 {
221 CacheItem(n);
222
223 wxHtmlCell *cell = m_cache->Get(n);
224 wxCHECK_RET( cell, _T("this cell should be cached!") );
225
226 wxHtmlRenderingInfo htmlRendInfo;
227
228 // draw the selected cell in selected state
229 if ( IsSelected(n) )
230 {
231 wxHtmlSelection htmlSel;
232 htmlSel.Set(wxPoint(0, 0), cell, wxPoint(INT_MAX, INT_MAX), cell);
233 htmlRendInfo.SetSelection(&htmlSel);
234 //htmlRendInfo.SetSelectionState(wxHTML_SEL_IN);
235 }
236
237 // note that we can't stop drawing exactly at the window boundary as then
238 // even the visible cells part could be not drawn, so always draw the
239 // entire cell
240 cell->Draw(dc, rect.x, rect.y, 0, INT_MAX, htmlRendInfo);
241 }
242
243 wxCoord wxHtmlListBox::OnMeasureItem(size_t n) const
244 {
245 CacheItem(n);
246
247 wxHtmlCell *cell = m_cache->Get(n);
248 wxCHECK_MSG( cell, 0, _T("this cell should be cached!") );
249
250 return cell->GetHeight() + cell->GetDescent();
251 }
252