| // Copyright 2016 PDFium Authors. All rights reserved. |
| // Use of this source code is governed by a BSD-style license that can be |
| // found in the LICENSE file. |
| |
| // Original code copyright 2014 Foxit Software Inc. http://www.foxitsoftware.com |
| |
| #include "xfa/fxfa/parser/cxfa_layoutitem.h" |
| |
| #include "xfa/fxfa/cxfa_ffnotify.h" |
| #include "xfa/fxfa/parser/cxfa_containerlayoutitem.h" |
| #include "xfa/fxfa/parser/cxfa_contentlayoutitem.h" |
| #include "xfa/fxfa/parser/cxfa_measurement.h" |
| #include "xfa/fxfa/parser/cxfa_node.h" |
| |
| void XFA_ReleaseLayoutItem(CXFA_LayoutItem* pLayoutItem) { |
| CXFA_LayoutItem* pNode = pLayoutItem->m_pFirstChild; |
| CXFA_FFNotify* pNotify = pLayoutItem->m_pFormNode->GetDocument()->GetNotify(); |
| CXFA_LayoutProcessor* pDocLayout = |
| pLayoutItem->m_pFormNode->GetDocument()->GetDocLayout(); |
| while (pNode) { |
| CXFA_LayoutItem* pNext = pNode->m_pNextSibling; |
| pNode->m_pParent = nullptr; |
| pNotify->OnLayoutItemRemoving(pDocLayout, pNode); |
| XFA_ReleaseLayoutItem(pNode); |
| pNode = pNext; |
| } |
| pNotify->OnLayoutItemRemoving(pDocLayout, pLayoutItem); |
| if (pLayoutItem->m_pFormNode->GetElementType() == XFA_Element::PageArea) { |
| pNotify->OnPageEvent(static_cast<CXFA_ContainerLayoutItem*>(pLayoutItem), |
| XFA_PAGEVIEWEVENT_PostRemoved); |
| } |
| delete pLayoutItem; |
| } |
| |
| CXFA_LayoutItem::CXFA_LayoutItem(CXFA_Node* pNode, bool bIsContentLayoutItem) |
| : m_pFormNode(pNode), |
| m_pParent(nullptr), |
| m_pNextSibling(nullptr), |
| m_pFirstChild(nullptr), |
| m_bIsContentLayoutItem(bIsContentLayoutItem) {} |
| |
| CXFA_LayoutItem::~CXFA_LayoutItem() {} |
| |
| CXFA_ContainerLayoutItem* CXFA_LayoutItem::AsContainerLayoutItem() { |
| return IsContainerLayoutItem() ? static_cast<CXFA_ContainerLayoutItem*>(this) |
| : nullptr; |
| } |
| |
| CXFA_ContentLayoutItem* CXFA_LayoutItem::AsContentLayoutItem() { |
| return IsContentLayoutItem() ? static_cast<CXFA_ContentLayoutItem*>(this) |
| : nullptr; |
| } |
| |
| CXFA_ContainerLayoutItem* CXFA_LayoutItem::GetPage() const { |
| for (CXFA_LayoutItem* pCurNode = const_cast<CXFA_LayoutItem*>(this); pCurNode; |
| pCurNode = pCurNode->m_pParent) { |
| if (pCurNode->m_pFormNode->GetElementType() == XFA_Element::PageArea) |
| return static_cast<CXFA_ContainerLayoutItem*>(pCurNode); |
| } |
| return nullptr; |
| } |
| |
| CFX_RectF CXFA_LayoutItem::GetRect(bool bRelative) const { |
| ASSERT(m_bIsContentLayoutItem); |
| |
| auto* pThis = static_cast<const CXFA_ContentLayoutItem*>(this); |
| CFX_PointF sPos = pThis->m_sPos; |
| CFX_SizeF sSize = pThis->m_sSize; |
| if (bRelative) |
| return CFX_RectF(sPos, sSize); |
| |
| for (CXFA_LayoutItem* pLayoutItem = pThis->m_pParent; pLayoutItem; |
| pLayoutItem = pLayoutItem->m_pParent) { |
| if (CXFA_ContentLayoutItem* pContent = pLayoutItem->AsContentLayoutItem()) { |
| sPos += pContent->m_sPos; |
| CXFA_Node* pMarginNode = |
| pLayoutItem->m_pFormNode->GetFirstChildByClass(XFA_Element::Margin); |
| if (pMarginNode) { |
| sPos += CFX_PointF(pMarginNode->GetMeasure(XFA_ATTRIBUTE_LeftInset) |
| .ToUnit(XFA_UNIT_Pt), |
| pMarginNode->GetMeasure(XFA_ATTRIBUTE_TopInset) |
| .ToUnit(XFA_UNIT_Pt)); |
| } |
| continue; |
| } |
| |
| if (pLayoutItem->m_pFormNode->GetElementType() == |
| XFA_Element::ContentArea) { |
| sPos += CFX_PointF(pLayoutItem->m_pFormNode->GetMeasure(XFA_ATTRIBUTE_X) |
| .ToUnit(XFA_UNIT_Pt), |
| pLayoutItem->m_pFormNode->GetMeasure(XFA_ATTRIBUTE_Y) |
| .ToUnit(XFA_UNIT_Pt)); |
| break; |
| } |
| if (pLayoutItem->m_pFormNode->GetElementType() == XFA_Element::PageArea) |
| break; |
| } |
| return CFX_RectF(sPos, sSize); |
| } |
| |
| CXFA_LayoutItem* CXFA_LayoutItem::GetFirst() { |
| ASSERT(m_bIsContentLayoutItem); |
| CXFA_ContentLayoutItem* pCurNode = static_cast<CXFA_ContentLayoutItem*>(this); |
| while (pCurNode->m_pPrev) |
| pCurNode = pCurNode->m_pPrev; |
| |
| return pCurNode; |
| } |
| |
| const CXFA_LayoutItem* CXFA_LayoutItem::GetLast() const { |
| ASSERT(m_bIsContentLayoutItem); |
| const CXFA_ContentLayoutItem* pCurNode = |
| static_cast<const CXFA_ContentLayoutItem*>(this); |
| while (pCurNode->m_pNext) |
| pCurNode = pCurNode->m_pNext; |
| |
| return pCurNode; |
| } |
| |
| CXFA_LayoutItem* CXFA_LayoutItem::GetPrev() const { |
| ASSERT(m_bIsContentLayoutItem); |
| |
| return static_cast<const CXFA_ContentLayoutItem*>(this)->m_pPrev; |
| } |
| |
| CXFA_LayoutItem* CXFA_LayoutItem::GetNext() const { |
| ASSERT(m_bIsContentLayoutItem); |
| return static_cast<const CXFA_ContentLayoutItem*>(this)->m_pNext; |
| } |
| |
| int32_t CXFA_LayoutItem::GetIndex() const { |
| ASSERT(m_bIsContentLayoutItem); |
| int32_t iIndex = 0; |
| const CXFA_ContentLayoutItem* pCurNode = |
| static_cast<const CXFA_ContentLayoutItem*>(this); |
| while (pCurNode->m_pPrev) { |
| pCurNode = pCurNode->m_pPrev; |
| ++iIndex; |
| } |
| return iIndex; |
| } |
| |
| int32_t CXFA_LayoutItem::GetCount() const { |
| ASSERT(m_bIsContentLayoutItem); |
| |
| int32_t iCount = GetIndex() + 1; |
| const CXFA_ContentLayoutItem* pCurNode = |
| static_cast<const CXFA_ContentLayoutItem*>(this); |
| while (pCurNode->m_pNext) { |
| pCurNode = pCurNode->m_pNext; |
| iCount++; |
| } |
| return iCount; |
| } |
| |
| void CXFA_LayoutItem::AddChild(CXFA_LayoutItem* pChildItem) { |
| if (pChildItem->m_pParent) |
| pChildItem->m_pParent->RemoveChild(pChildItem); |
| |
| pChildItem->m_pParent = this; |
| if (!m_pFirstChild) { |
| m_pFirstChild = pChildItem; |
| return; |
| } |
| |
| CXFA_LayoutItem* pExistingChildItem = m_pFirstChild; |
| while (pExistingChildItem->m_pNextSibling) |
| pExistingChildItem = pExistingChildItem->m_pNextSibling; |
| |
| pExistingChildItem->m_pNextSibling = pChildItem; |
| } |
| |
| void CXFA_LayoutItem::AddHeadChild(CXFA_LayoutItem* pChildItem) { |
| if (pChildItem->m_pParent) |
| pChildItem->m_pParent->RemoveChild(pChildItem); |
| |
| pChildItem->m_pParent = this; |
| if (!m_pFirstChild) { |
| m_pFirstChild = pChildItem; |
| return; |
| } |
| |
| CXFA_LayoutItem* pExistingChildItem = m_pFirstChild; |
| m_pFirstChild = pChildItem; |
| m_pFirstChild->m_pNextSibling = pExistingChildItem; |
| } |
| |
| void CXFA_LayoutItem::InsertChild(CXFA_LayoutItem* pBeforeItem, |
| CXFA_LayoutItem* pChildItem) { |
| if (pBeforeItem->m_pParent != this) |
| return; |
| if (pChildItem->m_pParent) |
| pChildItem->m_pParent = nullptr; |
| |
| pChildItem->m_pParent = this; |
| |
| CXFA_LayoutItem* pExistingChildItem = pBeforeItem->m_pNextSibling; |
| pBeforeItem->m_pNextSibling = pChildItem; |
| pChildItem->m_pNextSibling = pExistingChildItem; |
| } |
| |
| void CXFA_LayoutItem::RemoveChild(CXFA_LayoutItem* pChildItem) { |
| if (pChildItem->m_pParent != this) |
| return; |
| |
| if (m_pFirstChild == pChildItem) { |
| m_pFirstChild = pChildItem->m_pNextSibling; |
| } else { |
| CXFA_LayoutItem* pExistingChildItem = m_pFirstChild; |
| while (pExistingChildItem && |
| pExistingChildItem->m_pNextSibling != pChildItem) { |
| pExistingChildItem = pExistingChildItem->m_pNextSibling; |
| } |
| if (pExistingChildItem) |
| pExistingChildItem->m_pNextSibling = pChildItem->m_pNextSibling; |
| } |
| pChildItem->m_pNextSibling = nullptr; |
| pChildItem->m_pParent = nullptr; |
| } |