1 /* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- */
3 * This file is part of the LibreOffice project.
5 * This Source Code Form is subject to the terms of the Mozilla Public
6 * License, v. 2.0. If a copy of the MPL was not distributed with this
7 * file, You can obtain one at http://mozilla.org/MPL/2.0/.
9 * This file incorporates work covered by the following license notice:
11 * Licensed to the Apache Software Foundation (ASF) under one or more
12 * contributor license agreements. See the NOTICE file distributed
13 * with this work for additional information regarding copyright
14 * ownership. The ASF licenses this file to you under the Apache
15 * License, Version 2.0 (the "License"); you may not use this file
16 * except in compliance with the License. You may obtain a copy of
17 * the License at http://www.apache.org/licenses/LICENSE-2.0 .
20 #ifndef INCLUDED_SW_INC_HTMLTBL_HXX
21 #define INCLUDED_SW_INC_HTMLTBL_HXX
24 #include <vcl/timer.hxx>
25 #include <editeng/svxenum.hxx>
27 #include "swtypes.hxx"
32 class SwHTMLTableLayout
;
36 #define HTMLTABLE_RESIZE_NOW (ULONG_MAX)
38 class SwHTMLTableLayoutCnts
40 std::shared_ptr
<SwHTMLTableLayoutCnts
> m_xNext
; ///< The next content.
42 /// Only one of the following two pointers may be set!
43 SwTableBox
*m_pBox
; ///< A Box.
44 std::shared_ptr
<SwHTMLTableLayout
> m_xTable
; ///< A "table within a table".
46 /** During first run there are still no boxes. In this case
47 pStartNode is used instead of pBox. */
48 const SwStartNode
*m_pStartNode
;
50 /** The following counters indicate how often a pass has been
51 done for this content. Therefore they are compared against
52 a reference value. If 255 is reached the continue with 0.
53 This avoids reinitialization on every resize. */
54 sal_uInt8 m_nPass1Done
; ///< How many times has Pass 1 been called?
55 sal_uInt8 m_nWidthSet
; ///< How many times has the width been set?
57 bool m_bNoBreakTag
; ///< <NOBR>-Tag over complete content.
61 SwHTMLTableLayoutCnts(const SwStartNode
* pSttNd
, std::shared_ptr
<SwHTMLTableLayout
> const& rTab
,
62 bool bNoBreakTag
, std::shared_ptr
<SwHTMLTableLayoutCnts
> const& rNxt
);
64 void SetTableBox( SwTableBox
*pBx
) { m_pBox
= pBx
; }
65 SwTableBox
*GetTableBox() const { return m_pBox
; }
67 SwHTMLTableLayout
*GetTable() const { return m_xTable
.get(); }
69 const SwStartNode
*GetStartNode() const;
71 /// Calculation of next node.
72 const std::shared_ptr
<SwHTMLTableLayoutCnts
>& GetNext() const { return m_xNext
; }
74 void SetWidthSet( sal_uInt8 nRef
) { m_nWidthSet
= nRef
; }
75 bool IsWidthSet( sal_uInt8 nRef
) const { return nRef
==m_nWidthSet
; }
77 void SetPass1Done( sal_uInt8 nRef
) { m_nPass1Done
= nRef
; }
78 bool IsPass1Done( sal_uInt8 nRef
) const { return nRef
==m_nPass1Done
; }
80 bool HasNoBreakTag() const { return m_bNoBreakTag
; }
83 class SwHTMLTableLayoutCell
85 std::shared_ptr
<SwHTMLTableLayoutCnts
> m_xContents
; ///< Content of cell.
87 sal_uInt16 m_nRowSpan
; ///< ROWSPAN of cell.
88 sal_uInt16 m_nColSpan
; ///< COLSPAN of cell.
89 sal_uInt16 m_nWidthOption
; ///< Given width of cell in Twip or %.
91 bool m_bPercentWidthOption
: 1; ///< nWidth is %-value.
92 bool m_bNoWrapOption
: 1; ///< NOWRAP-option.
96 SwHTMLTableLayoutCell(std::shared_ptr
<SwHTMLTableLayoutCnts
> const& rCnts
,
97 sal_uInt16 nRSpan
, sal_uInt16 nCSpan
,
98 sal_uInt16 nWidthOpt
, bool bPercentWidthOpt
,
101 /// Set or get content of a cell.
102 void SetContents(std::shared_ptr
<SwHTMLTableLayoutCnts
> const& rCnts
) { m_xContents
= rCnts
; }
103 const std::shared_ptr
<SwHTMLTableLayoutCnts
>& GetContents() const { return m_xContents
; }
105 inline void SetProtected();
107 /// Set or get ROWSPAN/COLSPAN of cell.
108 void SetRowSpan( sal_uInt16 nRSpan
) { m_nRowSpan
= nRSpan
; }
109 sal_uInt16
GetRowSpan() const { return m_nRowSpan
; }
110 sal_uInt16
GetColSpan() const { return m_nColSpan
; }
112 sal_uInt16
GetWidthOption() const { return m_nWidthOption
; }
113 bool IsPercentWidthOption() const { return m_bPercentWidthOption
; }
115 bool HasNoWrapOption() const { return m_bNoWrapOption
; }
118 class SwHTMLTableLayoutColumn
121 /// Interim values of AutoLayoutPass1,
122 sal_uLong m_nMinNoAlign
, m_nMaxNoAlign
, m_nAbsMinNoAlign
;
124 /// Results of AutoLayoutPass1
125 sal_uLong m_nMin
, m_nMax
;
127 /// Results of Pass 2.
128 sal_uInt16 m_nAbsColWidth
; ///< In Twips.
129 sal_uInt16 m_nRelColWidth
; ///< In Twips or relative to USHRT_MAX.
131 sal_uInt16 m_nWidthOption
; ///< Options of <COL> or <TD>/<TH>.
133 bool m_bRelWidthOption
: 1;
134 bool m_bLeftBorder
: 1;
138 SwHTMLTableLayoutColumn( sal_uInt16 nColWidthOpt
, bool bRelColWidthOpt
,
141 inline void MergeCellWidthOption( sal_uInt16 nWidth
, bool bPercent
);
142 inline void SetWidthOption( sal_uInt16 nWidth
);
144 sal_uInt16
GetWidthOption() const { return m_nWidthOption
; }
145 bool IsRelWidthOption() const { return m_bRelWidthOption
; }
147 inline void MergeMinMaxNoAlign( sal_uLong nMin
, sal_uLong nMax
, sal_uLong nAbsMin
);
148 sal_uLong
GetMinNoAlign() const { return m_nMinNoAlign
; }
149 sal_uLong
GetMaxNoAlign() const { return m_nMaxNoAlign
; }
150 sal_uLong
GetAbsMinNoAlign() const { return m_nAbsMinNoAlign
; }
151 inline void ClearPass1Info( bool bWidthOpt
);
153 inline void SetMinMax( sal_uLong nMin
, sal_uLong nMax
);
154 void SetMax( sal_uLong nVal
) { m_nMax
= nVal
; }
155 void AddToMin( sal_uLong nVal
) { m_nMin
+= nVal
; }
156 void AddToMax( sal_uLong nVal
) { m_nMax
+= nVal
; }
157 sal_uLong
GetMin() const { return m_nMin
; }
158 sal_uLong
GetMax() const { return m_nMax
; }
160 void SetAbsColWidth( sal_uInt16 nWidth
) { m_nAbsColWidth
= nWidth
; }
161 sal_uInt16
GetAbsColWidth() const { return m_nAbsColWidth
; }
163 void SetRelColWidth( sal_uInt16 nWidth
) { m_nRelColWidth
= nWidth
; }
164 sal_uInt16
GetRelColWidth() const { return m_nRelColWidth
; }
166 bool HasLeftBorder() const { return m_bLeftBorder
; }
169 class SwHTMLTableLayout
171 Timer m_aResizeTimer
; ///< Timer for DelayedResize.
173 std::vector
<std::unique_ptr
<SwHTMLTableLayoutColumn
>> m_aColumns
;
174 std::vector
<std::unique_ptr
<SwHTMLTableLayoutCell
>> m_aCells
;
176 const SwTable
*m_pSwTable
; ///< SwTable (Top-Table only).
178 sal_uLong m_nMin
; ///< Minimal width of table (Twips).
179 sal_uLong m_nMax
; ///< Maximal width of table (Twips).
181 sal_uInt16 m_nRows
; ///< Row count.
182 sal_uInt16 m_nCols
; ///< Column count.
184 sal_uInt16 m_nLeftMargin
; ///< Space to left margin (from paragraph).
185 sal_uInt16 m_nRightMargin
; ///< Space to left margin (from paragraph).
187 sal_uInt16 m_nInhAbsLeftSpace
; ///< Space inherited from surrounding box
188 sal_uInt16 m_nInhAbsRightSpace
; ///< that was added to boxes.
190 sal_uInt16 m_nRelLeftFill
; ///< Width of boxes relative to alignment
191 sal_uInt16 m_nRelRightFill
; ///< of tables in tables.
193 sal_uInt16 m_nRelTabWidth
; ///< Relative width of table.
195 sal_uInt16 m_nWidthOption
; ///< Width of table (in Twips or %).
196 sal_uInt16 m_nCellPadding
; ///< Space to contents (in Twips).
197 sal_uInt16 m_nCellSpacing
; ///< Cell spacing (in Twips).
198 sal_uInt16 m_nBorder
; /** Line strength of outer border, or rather the
199 space needed for it as calculated by Netscape. */
201 SwTwips m_nLeftBorderWidth
;
202 SwTwips m_nRightBorderWidth
;
203 sal_uInt16 m_nInhLeftBorderWidth
;
204 sal_uInt16 m_nInhRightBorderWidth
;
205 SwTwips m_nBorderWidth
;
207 sal_uInt16 m_nDelayedResizeAbsAvail
; ///< Param for delayed Resize.
208 sal_uInt16 m_nLastResizeAbsAvail
;
210 sal_uInt8 m_nPass1Done
; ///< Reference-values for
211 sal_uInt8 m_nWidthSet
; ///< the runs through loop.
213 SvxAdjust m_eTableAdjust
; ///< Alignment of table.
215 bool m_bColsOption
: 1; ///< Table has a COLS-option.
216 bool m_bColTags
: 1; ///< Table has COL/COLGRP tags.
217 bool m_bPercentWidthOption
: 1; ///< Width is given in percent.
218 bool m_bUseRelWidth
: 1; ///< SwTable gets relative width.
220 bool m_bMustResize
: 1; ///< Table width must be defined.
221 bool m_bExportable
: 1; ///< Layout may be used for export.
222 bool m_bBordersChanged
: 1; ///< Borders have been changed.
223 bool m_bMayBeInFlyFrame
: 1; ///< Table could be within frame.
225 bool m_bDelayedResizeRecalc
: 1; ///< Param for delayed Resize.
226 bool m_bMustNotResize
: 1; ///< Table may not be resized.
227 bool m_bMustNotRecalc
: 1; ///< Table may not be adapted to its contents.
229 void AddBorderWidth( sal_uLong
&rMin
, sal_uLong
&rMax
, sal_uLong
& rAbsMin
,
230 sal_uInt16 nCol
, sal_uInt16 nColSpan
,
231 bool bSwBorders
=true ) const;
232 void SetBoxWidth( SwTableBox
*pBox
, sal_uInt16 nCol
, sal_uInt16 nColSpan
) const;
234 const SwStartNode
*GetAnyBoxStartNode() const;
235 SwFrameFormat
*FindFlyFrameFormat() const;
236 const SwDoc
& GetDoc() const { return GetAnyBoxStartNode()->GetDoc(); }
238 void Resize_( sal_uInt16 nAbsAvail
, bool bRecalc
);
240 DECL_LINK( DelayedResize_Impl
, Timer
*, void );
242 static sal_uInt16
GetBrowseWidthByVisArea( const SwDoc
& rDoc
);
245 SwHTMLTableLayout( const SwTable
*pSwTable
,
246 sal_uInt16 nRows
, sal_uInt16 nCols
, bool bColsOpt
, bool ColTgs
,
247 sal_uInt16 nWidth
, bool bPercentWidth
, sal_uInt16 nBorderOpt
,
248 sal_uInt16 nCellPad
, sal_uInt16 nCellSp
, SvxAdjust eAdjust
,
249 sal_uInt16 nLMargin
, sal_uInt16 nRMargin
, sal_uInt16 nBWidth
,
250 sal_uInt16 nLeftBWidth
, sal_uInt16 nRightBWidth
);
252 ~SwHTMLTableLayout();
254 sal_uInt16
GetLeftCellSpace( sal_uInt16 nCol
, sal_uInt16 nColSpan
,
255 bool bSwBorders
=true ) const;
256 sal_uInt16
GetRightCellSpace( sal_uInt16 nCol
, sal_uInt16 nColSpan
,
257 bool bSwBorders
=true ) const;
258 inline sal_uInt16
GetInhCellSpace( sal_uInt16 nCol
, sal_uInt16 nColSpan
) const;
260 inline void SetInhBorderWidths( sal_uInt16 nLeft
, sal_uInt16 nRight
);
262 void GetAvail( sal_uInt16 nCol
, sal_uInt16 nColSpan
, sal_uInt16
& rAbsAvail
,
263 sal_uInt16
& rRelAvail
) const;
265 void AutoLayoutPass1();
266 void AutoLayoutPass2( sal_uInt16 nAbsAvail
, sal_uInt16 nRelAvail
,
267 sal_uInt16 nAbsLeftSpace
, sal_uInt16 nAbsRightSpace
,
268 sal_uInt16 nParentInhSpace
);
269 void SetWidths( bool bCallPass2
=false, sal_uInt16 nAbsAvail
=0,
270 sal_uInt16 nRelAvail
=0, sal_uInt16 nAbsLeftSpace
=0,
271 sal_uInt16 nAbsRightSpace
=0,
272 sal_uInt16 nParentInhSpace
=0 );
274 inline SwHTMLTableLayoutColumn
*GetColumn( sal_uInt16 nCol
) const;
275 inline void SetColumn( std::unique_ptr
<SwHTMLTableLayoutColumn
> pCol
, sal_uInt16 nCol
);
277 inline SwHTMLTableLayoutCell
*GetCell( sal_uInt16 nRow
, sal_uInt16 nCol
) const;
278 inline void SetCell( std::unique_ptr
<SwHTMLTableLayoutCell
> pCell
, sal_uInt16 nRow
, sal_uInt16 nCol
);
280 sal_uLong
GetMin() const { return m_nMin
; }
281 sal_uLong
GetMax() const { return m_nMax
; }
283 inline tools::Long
GetBrowseWidthMin() const;
285 bool HasColsOption() const { return m_bColsOption
; }
286 bool HasColTags() const { return m_bColTags
; }
288 bool IsTopTable() const { return m_pSwTable
!= nullptr; }
290 void SetMustResize( bool bSet
) { m_bMustResize
= bSet
; }
291 void SetMustNotResize( bool bSet
) { m_bMustNotResize
= bSet
; }
292 void SetMustNotRecalc( bool bSet
) { m_bMustNotRecalc
= bSet
; }
294 /** Recalculation of table widths for available width that has been passed.
295 - If bRecalc is set, contents of boxes are included into calculation.
296 - If bForce is set, table will be recalculated even if this was
297 disallowed by SetMustNotResize.
298 - If nDelay > 0 the calculation is delayed accordingly. Resizing calls
299 occurring during delay-time are ignored, but the delay may be counted
300 under certain circumstances.
301 - If nDelay == HTMLTABLE_RESIZE_NOW, resize immediately and do not
302 consider any resize-calls that might possibly be in order.
303 - The return value indicates whether the table has been changed. */
304 bool Resize( sal_uInt16 nAbsAvail
, bool bRecalc
=false, bool bForce
=false,
305 sal_uLong nDelay
=0 );
307 void BordersChanged( sal_uInt16 nAbsAvail
);
309 /** Calculate available width. This works only if a layout or a
310 SwViewShell exists. Otherwise returns 0.
311 This is needed by HTML-filter because it doesn't have access to the layout.) */
312 static sal_uInt16
GetBrowseWidth( const SwDoc
& rDoc
);
314 /// Calculates available width by table-frame.
315 sal_uInt16
GetBrowseWidthByTabFrame( const SwTabFrame
& rTabFrame
) const;
317 /** Calculates available width by the table-frame or
318 static GetBrowseWidth if no layout exists. */
319 sal_uInt16
GetBrowseWidthByTable( const SwDoc
& rDoc
) const;
322 sal_uInt16
GetWidthOption() const { return m_nWidthOption
; }
323 bool HasPercentWidthOption() const { return m_bPercentWidthOption
; }
325 sal_uInt16
GetCellPadding() const { return m_nCellPadding
; }
326 sal_uInt16
GetCellSpacing() const { return m_nCellSpacing
; }
327 sal_uInt16
GetBorder() const { return m_nBorder
; }
329 sal_uInt16
GetRowCount() const { return m_nRows
; }
330 sal_uInt16
GetColCount() const { return m_nCols
; }
332 void SetExportable( bool bSet
) { m_bExportable
= bSet
; }
333 bool IsExportable() const { return m_bExportable
; }
335 bool HaveBordersChanged() const { return m_bBordersChanged
; }
337 void SetMayBeInFlyFrame( bool bSet
) { m_bMayBeInFlyFrame
= bSet
; }
338 bool MayBeInFlyFrame() const { return m_bMayBeInFlyFrame
; }
341 inline void SwHTMLTableLayoutCell::SetProtected()
348 inline void SwHTMLTableLayoutColumn::MergeMinMaxNoAlign( sal_uLong nCMin
,
349 sal_uLong nCMax
, sal_uLong nAbsMin
)
351 if( nCMin
> m_nMinNoAlign
)
352 m_nMinNoAlign
= nCMin
;
353 if( nCMax
> m_nMaxNoAlign
)
354 m_nMaxNoAlign
= nCMax
;
355 if( nAbsMin
> m_nAbsMinNoAlign
)
356 m_nAbsMinNoAlign
= nAbsMin
;
359 inline void SwHTMLTableLayoutColumn::ClearPass1Info( bool bWidthOpt
)
361 m_nMinNoAlign
= m_nMaxNoAlign
= m_nAbsMinNoAlign
= MINLAY
;
366 m_bRelWidthOption
= false;
370 inline void SwHTMLTableLayoutColumn::MergeCellWidthOption(
371 sal_uInt16 nWidth
, bool bRel
)
373 if( !m_nWidthOption
||
374 (bRel
==m_bRelWidthOption
&& m_nWidthOption
< nWidth
) )
376 m_nWidthOption
= nWidth
;
377 m_bRelWidthOption
= bRel
;
381 inline void SwHTMLTableLayoutColumn::SetMinMax( sal_uLong nMn
, sal_uLong nMx
)
387 inline sal_uInt16
SwHTMLTableLayout::GetInhCellSpace( sal_uInt16 nCol
,
388 sal_uInt16 nColSpan
) const
390 sal_uInt16 nSpace
= 0;
392 nSpace
= nSpace
+ m_nInhAbsLeftSpace
;
393 if( nCol
+nColSpan
==m_nCols
)
394 nSpace
= nSpace
+ m_nInhAbsRightSpace
;
399 inline SwHTMLTableLayoutColumn
*SwHTMLTableLayout::GetColumn( sal_uInt16 nCol
) const
401 return m_aColumns
[nCol
].get();
404 inline void SwHTMLTableLayoutColumn::SetWidthOption( sal_uInt16 nWidth
)
406 m_nWidthOption
= nWidth
;
407 m_bRelWidthOption
= true;
410 inline void SwHTMLTableLayout::SetColumn( std::unique_ptr
<SwHTMLTableLayoutColumn
> pCol
, sal_uInt16 nCol
)
412 m_aColumns
[nCol
] = std::move(pCol
);
415 inline SwHTMLTableLayoutCell
*SwHTMLTableLayout::GetCell( sal_uInt16 nRow
, sal_uInt16 nCol
) const
417 return m_aCells
[static_cast<size_t>(nRow
)*m_nCols
+nCol
].get();
420 inline void SwHTMLTableLayout::SetCell( std::unique_ptr
<SwHTMLTableLayoutCell
> pCell
,
421 sal_uInt16 nRow
, sal_uInt16 nCol
)
423 m_aCells
[static_cast<size_t>(nRow
)*m_nCols
+nCol
] = std::move(pCell
);
426 inline tools::Long
SwHTMLTableLayout::GetBrowseWidthMin() const
428 return static_cast<tools::Long
>( (!m_nWidthOption
|| m_bPercentWidthOption
) ? m_nMin
: m_nRelTabWidth
);
431 void SwHTMLTableLayout::SetInhBorderWidths( sal_uInt16 nLeft
, sal_uInt16 nRight
)
433 m_nInhLeftBorderWidth
= nLeft
;
434 m_nInhRightBorderWidth
= nRight
;
439 /* vim:set shiftwidth=4 softtabstop=4 expandtab: */