1 /////////////////////////////////////////////////////////////////////////////
2 // Name: src/html/m_tables.cpp
3 // Purpose: wxHtml module for tables
4 // Author: Vaclav Slavik
5 // Copyright: (c) 1999 Vaclav Slavik
6 // Licence: wxWindows licence
7 /////////////////////////////////////////////////////////////////////////////
15 #if wxUSE_HTML && wxUSE_STREAMS
18 #include "wx/wxcrtvararg.h"
22 #include "wx/html/forcelnk.h"
23 #include "wx/html/m_templ.h"
25 #include "wx/html/htmlcell.h"
27 FORCE_LINK_ME(m_tables
)
30 #define TABLE_BORDER_CLR_1 wxColour(0xC5, 0xC2, 0xC5)
31 #define TABLE_BORDER_CLR_2 wxColour(0x62, 0x61, 0x62)
34 //-----------------------------------------------------------------------------
36 //-----------------------------------------------------------------------------
42 // width of the column either in pixels or percents
43 // ('width' is the number, 'units' determines its meaning)
44 int minWidth
, maxWidth
;
45 // minimal/maximal column width. This is needed by HTML 4.0
46 // layout algorithm and can be determined by trying to
47 // layout table cells with width=1 and width=infinity
48 int leftpos
, pixwidth
, maxrealwidth
;
49 // temporary (depends on actual width of table)
61 wxHtmlContainerCell
*cont
;
63 int minheight
, valign
;
69 class wxHtmlTableCell
: public wxHtmlContainerCell
72 /* These are real attributes: */
74 // number of columns; rows
75 int m_NumCols
, m_NumRows
;
76 // array of column information
77 colStruct
*m_ColsInfo
;
78 // 2D array of all cells in the table : m_CellInfo[row][column]
79 cellStruct
**m_CellInfo
;
80 // spaces between cells
82 // cells internal indentation
86 /* ...and these are valid only when parsing the table: */
88 // number of actual column (ranging from 0..m_NumCols)
89 int m_ActualCol
, m_ActualRow
;
91 // default values (for table and row):
92 wxColour m_tBkg
, m_rBkg
;
93 wxString m_tValign
, m_rValign
;
99 wxHtmlTableCell(wxHtmlContainerCell
*parent
, const wxHtmlTag
& tag
, double pixel_scale
= 1.0);
100 virtual ~wxHtmlTableCell();
102 virtual void RemoveExtraSpacing(bool top
, bool bottom
);
104 virtual void Layout(int w
);
106 void AddRow(const wxHtmlTag
& tag
);
107 void AddCell(wxHtmlContainerCell
*cell
, const wxHtmlTag
& tag
);
109 const wxColour
& GetRowDefaultBackgroundColour() const { return m_rBkg
; }
112 // Reallocates memory to given number of cols/rows
113 // and changes m_NumCols/m_NumRows value to reflect this change
114 // NOTE! You CAN'T change m_NumCols/m_NumRows before calling this!!
115 void ReallocCols(int cols
);
116 void ReallocRows(int rows
);
118 // Computes minimal and maximal widths of columns. Needs to be called
119 // only once, before first Layout().
120 void ComputeMinMaxWidths();
122 wxDECLARE_NO_COPY_CLASS(wxHtmlTableCell
);
127 wxHtmlTableCell::wxHtmlTableCell(wxHtmlContainerCell
*parent
, const wxHtmlTag
& tag
, double pixel_scale
)
128 : wxHtmlContainerCell(parent
)
130 m_PixelScale
= pixel_scale
;
132 m_NumCols
= m_NumRows
= 0;
134 m_ActualCol
= m_ActualRow
= -1;
137 if (tag
.HasParam(wxT("BGCOLOR")))
139 tag
.GetParamAsColour(wxT("BGCOLOR"), &m_tBkg
);
141 SetBackgroundColour(m_tBkg
);
143 if (tag
.HasParam(wxT("VALIGN")))
144 m_tValign
= tag
.GetParam(wxT("VALIGN"));
146 m_tValign
= wxEmptyString
;
147 if (!tag
.GetParamAsInt(wxT("CELLSPACING"), &m_Spacing
))
149 if (!tag
.GetParamAsInt(wxT("CELLPADDING"), &m_Padding
))
151 m_Spacing
= (int)(m_PixelScale
* (double)m_Spacing
);
152 m_Padding
= (int)(m_PixelScale
* (double)m_Padding
);
154 if(tag
.HasParam(wxT("BORDER")))
156 if(tag
.GetParam("BORDER").IsEmpty())
159 tag
.GetParamAsInt(wxT("BORDER"), &m_Border
);
162 SetBorder(TABLE_BORDER_CLR_1
, TABLE_BORDER_CLR_2
, m_Border
); // special case see wxHtmlContainerCell::Draw
163 else if (m_Border
> 0)
164 SetBorder(TABLE_BORDER_CLR_1
, TABLE_BORDER_CLR_2
, (int)(m_PixelScale
* (double)m_Border
));
172 wxHtmlTableCell::~wxHtmlTableCell()
174 if (m_ColsInfo
) free(m_ColsInfo
);
177 for (int i
= 0; i
< m_NumRows
; i
++)
184 void wxHtmlTableCell::RemoveExtraSpacing(bool WXUNUSED(top
),
185 bool WXUNUSED(bottom
))
187 // Don't remove any spacing in the table -- it's always desirable,
188 // because it's part of table's definition.
189 // (If wxHtmlContainerCell::RemoveExtraSpacing() was applied to tables,
190 // then upper left cell of a table would be positioned above other cells
191 // if the table was the first element on the page.)
194 void wxHtmlTableCell::ReallocCols(int cols
)
198 for (i
= 0; i
< m_NumRows
; i
++)
200 m_CellInfo
[i
] = (cellStruct
*) realloc(m_CellInfo
[i
], sizeof(cellStruct
) * cols
);
201 for (j
= m_NumCols
; j
< cols
; j
++)
202 m_CellInfo
[i
][j
].flag
= cellFree
;
205 m_ColsInfo
= (colStruct
*) realloc(m_ColsInfo
, sizeof(colStruct
) * cols
);
206 for (j
= m_NumCols
; j
< cols
; j
++)
208 m_ColsInfo
[j
].width
= 0;
209 m_ColsInfo
[j
].units
= wxHTML_UNITS_PERCENT
;
210 m_ColsInfo
[j
].minWidth
= m_ColsInfo
[j
].maxWidth
= -1;
218 void wxHtmlTableCell::ReallocRows(int rows
)
220 m_CellInfo
= (cellStruct
**) realloc(m_CellInfo
, sizeof(cellStruct
*) * rows
);
221 for (int row
= m_NumRows
; row
< rows
; row
++)
224 m_CellInfo
[row
] = NULL
;
227 m_CellInfo
[row
] = (cellStruct
*) malloc(sizeof(cellStruct
) * m_NumCols
);
228 for (int col
= 0; col
< m_NumCols
; col
++)
229 m_CellInfo
[row
][col
].flag
= cellFree
;
236 void wxHtmlTableCell::AddRow(const wxHtmlTag
& tag
)
239 // VS: real allocation of row entry is done in AddCell in order
240 // to correctly handle empty rows (i.e. "<tr></tr>")
241 // m_ActualCol == -1 indicates that AddCell has to allocate new row.
245 if (tag
.HasParam(wxT("BGCOLOR")))
246 tag
.GetParamAsColour(wxT("BGCOLOR"), &m_rBkg
);
247 if (tag
.HasParam(wxT("VALIGN")))
248 m_rValign
= tag
.GetParam(wxT("VALIGN"));
250 m_rValign
= m_tValign
;
255 void wxHtmlTableCell::AddCell(wxHtmlContainerCell
*cell
, const wxHtmlTag
& tag
)
257 // Is this cell in new row?
258 // VS: we can't do it in AddRow, see my comment there
259 if (m_ActualCol
== -1)
261 if (m_ActualRow
+ 1 > m_NumRows
- 1)
262 ReallocRows(m_ActualRow
+ 2);
270 } while ((m_ActualCol
< m_NumCols
) &&
271 (m_CellInfo
[m_ActualRow
][m_ActualCol
].flag
!= cellFree
));
273 if (m_ActualCol
> m_NumCols
- 1)
274 ReallocCols(m_ActualCol
+ 1);
276 int r
= m_ActualRow
, c
= m_ActualCol
;
278 m_CellInfo
[r
][c
].cont
= cell
;
279 m_CellInfo
[r
][c
].colspan
= 1;
280 m_CellInfo
[r
][c
].rowspan
= 1;
281 m_CellInfo
[r
][c
].flag
= cellUsed
;
282 m_CellInfo
[r
][c
].minheight
= 0;
283 m_CellInfo
[r
][c
].valign
= wxHTML_ALIGN_TOP
;
285 /* scan for parameters: */
289 if (tag
.GetParamAsString(wxT("ID"), &idvalue
))
291 cell
->SetId(idvalue
);
296 if (tag
.HasParam(wxT("WIDTH")))
298 wxString wd
= tag
.GetParam(wxT("WIDTH"));
300 if (!wd
.empty() && wd
[wd
.length()-1] == wxT('%'))
302 if ( wxSscanf(wd
.c_str(), wxT("%i%%"), &m_ColsInfo
[c
].width
) == 1 )
304 m_ColsInfo
[c
].units
= wxHTML_UNITS_PERCENT
;
310 if ( wd
.ToLong(&width
) )
312 m_ColsInfo
[c
].width
= (int)(m_PixelScale
* (double)width
);
313 m_ColsInfo
[c
].units
= wxHTML_UNITS_PIXELS
;
322 tag
.GetParamAsInt(wxT("COLSPAN"), &m_CellInfo
[r
][c
].colspan
);
323 tag
.GetParamAsInt(wxT("ROWSPAN"), &m_CellInfo
[r
][c
].rowspan
);
325 // VS: the standard says this about col/rowspan:
326 // "This attribute specifies the number of rows spanned by the
327 // current cell. The default value of this attribute is one ("1").
328 // The value zero ("0") means that the cell spans all rows from the
329 // current row to the last row of the table." All mainstream
330 // browsers act as if 0==1, though, and so does wxHTML.
331 if (m_CellInfo
[r
][c
].colspan
< 1)
332 m_CellInfo
[r
][c
].colspan
= 1;
333 if (m_CellInfo
[r
][c
].rowspan
< 1)
334 m_CellInfo
[r
][c
].rowspan
= 1;
336 if ((m_CellInfo
[r
][c
].colspan
> 1) || (m_CellInfo
[r
][c
].rowspan
> 1))
340 if (r
+ m_CellInfo
[r
][c
].rowspan
> m_NumRows
)
341 ReallocRows(r
+ m_CellInfo
[r
][c
].rowspan
);
342 if (c
+ m_CellInfo
[r
][c
].colspan
> m_NumCols
)
343 ReallocCols(c
+ m_CellInfo
[r
][c
].colspan
);
344 for (i
= r
; i
< r
+ m_CellInfo
[r
][c
].rowspan
; i
++)
345 for (j
= c
; j
< c
+ m_CellInfo
[r
][c
].colspan
; j
++)
346 m_CellInfo
[i
][j
].flag
= cellSpan
;
347 m_CellInfo
[r
][c
].flag
= cellUsed
;
353 wxColour bk
= m_rBkg
;
354 if (tag
.HasParam(wxT("BGCOLOR")))
355 tag
.GetParamAsColour(wxT("BGCOLOR"), &bk
);
357 cell
->SetBackgroundColour(bk
);
360 cell
->SetBorder(TABLE_BORDER_CLR_2
, TABLE_BORDER_CLR_1
);
362 // vertical alignment:
365 if (tag
.HasParam(wxT("VALIGN")))
366 valign
= tag
.GetParam(wxT("VALIGN"));
370 if (valign
== wxT("TOP"))
371 m_CellInfo
[r
][c
].valign
= wxHTML_ALIGN_TOP
;
372 else if (valign
== wxT("BOTTOM"))
373 m_CellInfo
[r
][c
].valign
= wxHTML_ALIGN_BOTTOM
;
374 else m_CellInfo
[r
][c
].valign
= wxHTML_ALIGN_CENTER
;
378 if (tag
.HasParam(wxT("NOWRAP")))
379 m_CellInfo
[r
][c
].nowrap
= true;
381 m_CellInfo
[r
][c
].nowrap
= false;
383 cell
->SetIndent(m_Padding
, wxHTML_INDENT_ALL
, wxHTML_UNITS_PIXELS
);
386 void wxHtmlTableCell::ComputeMinMaxWidths()
388 if (m_NumCols
== 0 || m_ColsInfo
[0].minWidth
!= wxDefaultCoord
) return;
392 for (int c
= 0; c
< m_NumCols
; c
++)
394 for (int r
= 0; r
< m_NumRows
; r
++)
396 cellStruct
& cell
= m_CellInfo
[r
][c
];
397 if (cell
.flag
== cellUsed
)
399 cell
.cont
->Layout(2*m_Padding
+ 1);
400 int maxWidth
= cell
.cont
->GetMaxTotalWidth();
401 int width
= cell
.nowrap
?maxWidth
:cell
.cont
->GetWidth();
402 width
-= (cell
.colspan
-1) * m_Spacing
;
403 maxWidth
-= (cell
.colspan
-1) * m_Spacing
;
404 // HTML 4.0 says it is acceptable to distribute min/max
405 width
/= cell
.colspan
;
406 maxWidth
/= cell
.colspan
;
407 for (int j
= 0; j
< cell
.colspan
; j
++) {
408 if (width
> m_ColsInfo
[c
+j
].minWidth
)
409 m_ColsInfo
[c
+j
].minWidth
= width
;
410 if (maxWidth
> m_ColsInfo
[c
+j
].maxWidth
)
411 m_ColsInfo
[c
+j
].maxWidth
= maxWidth
;
415 // Calculate maximum table width, required for nested tables
416 if (m_ColsInfo
[c
].units
== wxHTML_UNITS_PIXELS
)
417 m_MaxTotalWidth
+= wxMax(m_ColsInfo
[c
].width
, m_ColsInfo
[c
].minWidth
);
418 else if ((m_ColsInfo
[c
].units
== wxHTML_UNITS_PERCENT
) && (m_ColsInfo
[c
].width
!= 0))
419 percentage
+= m_ColsInfo
[c
].width
;
421 m_MaxTotalWidth
+= m_ColsInfo
[c
].maxWidth
;
424 if (percentage
>= 100)
426 // Table would have infinite length
427 // Make it ridiculous large
428 m_MaxTotalWidth
= 0xFFFFFF;
431 m_MaxTotalWidth
= m_MaxTotalWidth
* 100 / (100 - percentage
);
433 m_MaxTotalWidth
+= (m_NumCols
+ 1) * m_Spacing
+ 2 * m_Border
;
436 void wxHtmlTableCell::Layout(int w
)
438 ComputeMinMaxWidths();
440 wxHtmlCell::Layout(w
);
448 if (m_WidthFloatUnits
== wxHTML_UNITS_PERCENT
)
450 if (m_WidthFloat
< 0)
452 if (m_WidthFloat
< -100)
454 m_Width
= (100 + m_WidthFloat
) * w
/ 100;
458 if (m_WidthFloat
> 100)
460 m_Width
= m_WidthFloat
* w
/ 100;
465 if (m_WidthFloat
< 0) m_Width
= w
+ m_WidthFloat
;
466 else m_Width
= m_WidthFloat
;
476 /* 1. setup columns widths:
478 The algorithm tries to keep the table size less than w if possible.
481 int wpix
= m_Width
- (m_NumCols
+ 1) * m_Spacing
- 2 * m_Border
; // Available space for cell content
484 // 1a. setup fixed-width columns:
485 for (i
= 0; i
< m_NumCols
; i
++)
486 if (m_ColsInfo
[i
].units
== wxHTML_UNITS_PIXELS
)
488 m_ColsInfo
[i
].pixwidth
= wxMax(m_ColsInfo
[i
].width
,
489 m_ColsInfo
[i
].minWidth
);
490 wpix
-= m_ColsInfo
[i
].pixwidth
;
493 // 1b. Calculate maximum possible width if line wrapping would be disabled
494 // Recalculate total width if m_WidthFloat is zero to keep tables as small
497 for (i
= 0; i
< m_NumCols
; i
++)
498 if (m_ColsInfo
[i
].width
== 0)
500 maxWidth
+= m_ColsInfo
[i
].maxWidth
;
505 // Recalculate table width since no table width was initially given
506 int newWidth
= m_Width
- wpix
+ maxWidth
;
508 // Make sure that floating-width columns will have the right size.
509 // Calculate sum of all floating-width columns
511 for (i
= 0; i
< m_NumCols
; i
++)
512 if ((m_ColsInfo
[i
].units
== wxHTML_UNITS_PERCENT
) && (m_ColsInfo
[i
].width
!= 0))
513 percentage
+= m_ColsInfo
[i
].width
;
515 if (percentage
>= 100)
518 newWidth
= newWidth
* 100 / (100 - percentage
);
520 newWidth
= wxMin(newWidth
, w
- (m_NumCols
+ 1) * m_Spacing
- 2 * m_Border
);
521 wpix
-= m_Width
- newWidth
;
526 // 1c. setup floating-width columns:
528 for (i
= 0; i
< m_NumCols
; i
++)
529 if ((m_ColsInfo
[i
].units
== wxHTML_UNITS_PERCENT
) && (m_ColsInfo
[i
].width
!= 0))
531 m_ColsInfo
[i
].pixwidth
= wxMin(m_ColsInfo
[i
].width
, 100) * wpix
/ 100;
533 // Make sure to leave enough space for the other columns
534 int minRequired
= m_Border
;
535 for (j
= 0; j
< m_NumCols
; j
++)
537 if ((m_ColsInfo
[j
].units
== wxHTML_UNITS_PERCENT
&& j
> i
) ||
538 !m_ColsInfo
[j
].width
)
539 minRequired
+= m_ColsInfo
[j
].minWidth
;
541 m_ColsInfo
[i
].pixwidth
= wxMax(wxMin(wtemp
- minRequired
, m_ColsInfo
[i
].pixwidth
), m_ColsInfo
[i
].minWidth
);
543 wtemp
-= m_ColsInfo
[i
].pixwidth
;
545 wpix
= wtemp
; // minimum cells width
547 // 1d. setup default columns (no width specification supplied):
548 // The algorithm assigns calculates the maximum possible width if line
549 // wrapping would be disabled and assigns column width as a fraction
550 // based upon the maximum width of a column
551 // FIXME: I'm not sure if this algorithm is conform to HTML standard,
552 // though it seems to be much better than the old one
554 for (i
= j
= 0; i
< m_NumCols
; i
++)
555 if (m_ColsInfo
[i
].width
== 0) j
++;
560 for (i
= 0; i
< m_NumCols
; i
++)
561 if (m_ColsInfo
[i
].width
== 0)
563 // Assign with, make sure not to drop below minWidth
565 m_ColsInfo
[i
].pixwidth
= (int)(wpix
* (m_ColsInfo
[i
].maxWidth
/ (float)maxWidth
) + 0.5);
567 m_ColsInfo
[i
].pixwidth
= wpix
/ j
;
569 // Make sure to leave enough space for the other columns
572 for (r
= i
+ 1; r
< m_NumCols
; r
++)
574 if (!m_ColsInfo
[r
].width
)
575 minRequired
+= m_ColsInfo
[r
].minWidth
;
577 m_ColsInfo
[i
].pixwidth
= wxMax(wxMin(wpix
- minRequired
, m_ColsInfo
[i
].pixwidth
), m_ColsInfo
[i
].minWidth
);
581 if (m_ColsInfo
[i
].pixwidth
> (wpix
* (m_ColsInfo
[i
].maxWidth
/ (float)maxWidth
) + 0.5))
583 int diff
= (int)(m_ColsInfo
[i
].pixwidth
- (wpix
* m_ColsInfo
[i
].maxWidth
/ (float)maxWidth
+ 0.5));
584 maxWidth
+= diff
- m_ColsInfo
[i
].maxWidth
;
587 maxWidth
-= m_ColsInfo
[i
].maxWidth
;
589 wpix
-= m_ColsInfo
[i
].pixwidth
;
593 /* 2. compute positions of columns: */
595 int wpos
= m_Spacing
+ m_Border
;
596 for (int i
= 0; i
< m_NumCols
; i
++)
598 m_ColsInfo
[i
].leftpos
= wpos
;
599 wpos
+= m_ColsInfo
[i
].pixwidth
+ m_Spacing
;
602 // add the remaining space to the last column
603 if (m_NumCols
> 0 && wpos
< m_Width
- m_Border
)
604 m_ColsInfo
[m_NumCols
-1].pixwidth
+= m_Width
- wpos
- m_Border
;
607 /* 3. sub-layout all cells: */
609 int *ypos
= new int[m_NumRows
+ 1];
613 wxHtmlContainerCell
*actcell
;
615 ypos
[0] = m_Spacing
+ m_Border
;
616 for (actrow
= 1; actrow
<= m_NumRows
; actrow
++) ypos
[actrow
] = -1;
617 for (actrow
= 0; actrow
< m_NumRows
; actrow
++)
619 if (ypos
[actrow
] == -1) ypos
[actrow
] = ypos
[actrow
-1];
620 // 3a. sub-layout and detect max height:
622 for (actcol
= 0; actcol
< m_NumCols
; actcol
++) {
623 if (m_CellInfo
[actrow
][actcol
].flag
!= cellUsed
) continue;
624 actcell
= m_CellInfo
[actrow
][actcol
].cont
;
626 for (int i
= actcol
; i
< m_CellInfo
[actrow
][actcol
].colspan
+ actcol
; i
++)
627 fullwid
+= m_ColsInfo
[i
].pixwidth
;
628 fullwid
+= (m_CellInfo
[actrow
][actcol
].colspan
- 1) * m_Spacing
;
629 actcell
->SetMinHeight(m_CellInfo
[actrow
][actcol
].minheight
, m_CellInfo
[actrow
][actcol
].valign
);
630 actcell
->Layout(fullwid
);
632 if (ypos
[actrow
] + actcell
->GetHeight() + m_CellInfo
[actrow
][actcol
].rowspan
* m_Spacing
> ypos
[actrow
+ m_CellInfo
[actrow
][actcol
].rowspan
])
633 ypos
[actrow
+ m_CellInfo
[actrow
][actcol
].rowspan
] =
634 ypos
[actrow
] + actcell
->GetHeight() + m_CellInfo
[actrow
][actcol
].rowspan
* m_Spacing
;
638 for (actrow
= 0; actrow
< m_NumRows
; actrow
++)
640 // 3b. place cells in row & let'em all have same height:
642 for (actcol
= 0; actcol
< m_NumCols
; actcol
++)
644 if (m_CellInfo
[actrow
][actcol
].flag
!= cellUsed
) continue;
645 actcell
= m_CellInfo
[actrow
][actcol
].cont
;
646 actcell
->SetMinHeight(
647 ypos
[actrow
+ m_CellInfo
[actrow
][actcol
].rowspan
] - ypos
[actrow
] - m_Spacing
,
648 m_CellInfo
[actrow
][actcol
].valign
);
650 for (int i
= actcol
; i
< m_CellInfo
[actrow
][actcol
].colspan
+ actcol
; i
++)
651 fullwid
+= m_ColsInfo
[i
].pixwidth
;
652 fullwid
+= (m_CellInfo
[actrow
][actcol
].colspan
- 1) * m_Spacing
;
653 actcell
->Layout(fullwid
);
654 actcell
->SetPos(m_ColsInfo
[actcol
].leftpos
, ypos
[actrow
]);
657 m_Height
= ypos
[m_NumRows
] + m_Border
;
661 /* 4. adjust table's width if it was too small: */
664 int twidth
= m_ColsInfo
[m_NumCols
-1].leftpos
+
665 m_ColsInfo
[m_NumCols
-1].pixwidth
+ m_Spacing
+ m_Border
;
666 if (twidth
> m_Width
)
676 //-----------------------------------------------------------------------------
677 // The tables handler:
678 //-----------------------------------------------------------------------------
681 TAG_HANDLER_BEGIN(TABLE
, "TABLE,TR,TD,TH")
684 wxHtmlTableCell
* m_Table
;
685 wxString m_tAlign
, m_rAlign
;
686 wxHtmlContainerCell
*m_enclosingContainer
;
688 // Call ParseInner() preserving background colour and mode after any
689 // changes done by it.
690 void CallParseInnerWithBg(const wxHtmlTag
& tag
, const wxColour
& colBg
)
692 const wxColour oldbackclr
= m_WParser
->GetActualBackgroundColor();
693 const int oldbackmode
= m_WParser
->GetActualBackgroundMode();
696 m_WParser
->SetActualBackgroundColor(colBg
);
697 m_WParser
->SetActualBackgroundMode(wxBRUSHSTYLE_SOLID
);
698 m_WParser
->GetContainer()->InsertCell(
699 new wxHtmlColourCell(colBg
, wxHTML_CLR_BACKGROUND
)
705 if ( oldbackmode
!= m_WParser
->GetActualBackgroundMode() ||
706 oldbackclr
!= m_WParser
->GetActualBackgroundColor() )
708 m_WParser
->SetActualBackgroundMode(oldbackmode
);
709 m_WParser
->SetActualBackgroundColor(oldbackclr
);
710 m_WParser
->GetContainer()->InsertCell(
711 new wxHtmlColourCell(oldbackclr
,
712 oldbackmode
== wxBRUSHSTYLE_TRANSPARENT
713 ? wxHTML_CLR_TRANSPARENT_BACKGROUND
714 : wxHTML_CLR_BACKGROUND
)
719 TAG_HANDLER_CONSTR(TABLE
)
722 m_enclosingContainer
= NULL
;
723 m_tAlign
= m_rAlign
= wxEmptyString
;
727 TAG_HANDLER_PROC(tag
)
729 wxHtmlContainerCell
*c
;
731 // new table started, backup upper-level table (if any) and create new:
732 if (tag
.GetName() == wxT("TABLE"))
734 wxHtmlTableCell
*oldt
= m_Table
;
736 wxHtmlContainerCell
*oldEnclosing
= m_enclosingContainer
;
737 m_enclosingContainer
= c
= m_WParser
->OpenContainer();
739 m_Table
= new wxHtmlTableCell(c
, tag
, m_WParser
->GetPixelScale());
743 if (tag
.HasParam(wxT("WIDTH")))
746 bool wpercent
= false;
747 if (tag
.GetParamAsIntOrPercent(wxT("WIDTH"), &width
, wpercent
))
751 m_Table
->SetWidthFloat(width
, wxHTML_UNITS_PERCENT
);
755 m_Table
->SetWidthFloat((int)(m_WParser
->GetPixelScale() * width
), wxHTML_UNITS_PIXELS
);
760 m_Table
->SetWidthFloat(0, wxHTML_UNITS_PIXELS
);
762 int oldAlign
= m_WParser
->GetAlign();
763 m_tAlign
= wxEmptyString
;
764 if (tag
.HasParam(wxT("ALIGN")))
765 m_tAlign
= tag
.GetParam(wxT("ALIGN"));
767 CallParseInnerWithBg(tag
, m_Table
->GetBackgroundColour());
769 m_WParser
->SetAlign(oldAlign
);
770 m_WParser
->SetContainer(m_enclosingContainer
);
771 m_WParser
->CloseContainer();
774 m_enclosingContainer
= oldEnclosing
;
776 return true; // ParseInner() called
783 if (tag
.GetName() == wxT("TR"))
785 m_Table
->AddRow(tag
);
787 if (tag
.HasParam(wxT("ALIGN")))
788 m_rAlign
= tag
.GetParam(wxT("ALIGN"));
794 c
= m_WParser
->SetContainer(new wxHtmlContainerCell(m_Table
));
795 m_Table
->AddCell(c
, tag
);
797 m_WParser
->OpenContainer();
799 const bool isHeader
= tag
.GetName() == wxT("TH");
802 if (tag
.HasParam(wxT("ALIGN")))
803 als
= tag
.GetParam(wxT("ALIGN"));
808 if (als
== wxT("RIGHT"))
809 m_WParser
->SetAlign(wxHTML_ALIGN_RIGHT
);
810 else if (als
== wxT("LEFT"))
811 m_WParser
->SetAlign(wxHTML_ALIGN_LEFT
);
812 else if (als
== wxT("CENTER"))
813 m_WParser
->SetAlign(wxHTML_ALIGN_CENTER
);
814 else // use default alignment
815 m_WParser
->SetAlign(isHeader
? wxHTML_ALIGN_CENTER
816 : wxHTML_ALIGN_LEFT
);
818 m_WParser
->OpenContainer();
820 // the header should be rendered in bold by default
824 boldOld
= m_WParser
->GetFontBold();
825 m_WParser
->SetFontBold(true);
826 m_WParser
->GetContainer()->InsertCell(
827 new wxHtmlFontCell(m_WParser
->CreateCurrentFont()));
831 if ( !tag
.GetParamAsColour(wxT("BGCOLOR"), &bgCol
) )
832 bgCol
= m_Table
->GetRowDefaultBackgroundColour();
834 CallParseInnerWithBg(tag
, bgCol
);
838 m_WParser
->SetFontBold(boldOld
);
839 m_WParser
->GetContainer()->InsertCell(
840 new wxHtmlFontCell(m_WParser
->CreateCurrentFont()));
843 // set the current container back to the enclosing one so that
844 // text outside of <th> and <td> isn't included in any cell
845 // (this happens often enough in practice because it's common
846 // to use whitespace between </td> and the next <td>):
847 m_WParser
->SetContainer(m_enclosingContainer
);
849 return true; // ParseInner() called
856 TAG_HANDLER_END(TABLE
)
862 TAGS_MODULE_BEGIN(Tables
)
864 TAGS_MODULE_ADD(TABLE
)
866 TAGS_MODULE_END(Tables
)