// Copyright 2016 The PDFium Authors // 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/layout/cxfa_layoutprocessor.h" #include "fxjs/gc/container_trace.h" #include "fxjs/xfa/cjx_object.h" #include "v8/include/cppgc/heap.h" #include "xfa/fxfa/layout/cxfa_contentlayoutitem.h" #include "xfa/fxfa/layout/cxfa_contentlayoutprocessor.h" #include "xfa/fxfa/layout/cxfa_viewlayoutprocessor.h" #include "xfa/fxfa/parser/cxfa_document.h" #include "xfa/fxfa/parser/cxfa_localemgr.h" #include "xfa/fxfa/parser/cxfa_measurement.h" #include "xfa/fxfa/parser/cxfa_node.h" #include "xfa/fxfa/parser/cxfa_subform.h" #include "xfa/fxfa/parser/xfa_document_datamerger_imp.h" #include "xfa/fxfa/parser/xfa_utils.h" // static CXFA_LayoutProcessor* CXFA_LayoutProcessor::FromDocument( const CXFA_Document* pXFADoc) { return static_cast(pXFADoc->GetLayoutProcessor()); } CXFA_LayoutProcessor::CXFA_LayoutProcessor(cppgc::Heap* pHeap) : m_pHeap(pHeap) {} CXFA_LayoutProcessor::~CXFA_LayoutProcessor() = default; void CXFA_LayoutProcessor::Trace(cppgc::Visitor* visitor) const { CXFA_Document::LayoutProcessorIface::Trace(visitor); visitor->Trace(m_pViewLayoutProcessor); visitor->Trace(m_pContentLayoutProcessor); } void CXFA_LayoutProcessor::SetForceRelayout() { m_bNeedLayout = true; } int32_t CXFA_LayoutProcessor::StartLayout() { return NeedLayout() ? RestartLayout() : 100; } int32_t CXFA_LayoutProcessor::RestartLayout() { m_pContentLayoutProcessor = nullptr; m_nProgressCounter = 0; CXFA_Node* pFormPacketNode = ToNode(GetDocument()->GetXFAObject(XFA_HASHCODE_Form)); if (!pFormPacketNode) return -1; CXFA_Subform* pFormRoot = pFormPacketNode->GetFirstChildByClass(XFA_Element::Subform); if (!pFormRoot) return -1; if (!m_pViewLayoutProcessor) { m_pViewLayoutProcessor = cppgc::MakeGarbageCollected( GetHeap()->GetAllocationHandle(), GetHeap(), this); } if (!m_pViewLayoutProcessor->InitLayoutPage(pFormRoot)) return -1; if (!m_pViewLayoutProcessor->PrepareFirstPage(pFormRoot)) return -1; m_pContentLayoutProcessor = cppgc::MakeGarbageCollected( GetHeap()->GetAllocationHandle(), GetHeap(), pFormRoot, m_pViewLayoutProcessor); m_nProgressCounter = 1; return 0; } int32_t CXFA_LayoutProcessor::DoLayout() { if (m_nProgressCounter < 1) return -1; CXFA_ContentLayoutProcessor::Result eStatus; CXFA_Node* pFormNode = m_pContentLayoutProcessor->GetFormNode(); float fPosX = pFormNode->JSObject()->GetMeasureInUnit(XFA_Attribute::X, XFA_Unit::Pt); float fPosY = pFormNode->JSObject()->GetMeasureInUnit(XFA_Attribute::Y, XFA_Unit::Pt); do { float fAvailHeight = m_pViewLayoutProcessor->GetAvailHeight(); eStatus = m_pContentLayoutProcessor->DoLayout(true, fAvailHeight, fAvailHeight); if (eStatus != CXFA_ContentLayoutProcessor::Result::kDone) m_nProgressCounter++; CXFA_ContentLayoutItem* pLayoutItem = m_pContentLayoutProcessor->ExtractLayoutItem(); if (pLayoutItem) pLayoutItem->m_sPos = CFX_PointF(fPosX, fPosY); m_pViewLayoutProcessor->SubmitContentItem(pLayoutItem, eStatus); } while (eStatus != CXFA_ContentLayoutProcessor::Result::kDone); if (eStatus == CXFA_ContentLayoutProcessor::Result::kDone) { m_pViewLayoutProcessor->FinishPaginatedPageSets(); m_pViewLayoutProcessor->SyncLayoutData(); m_bHasChangedContainers = false; m_bNeedLayout = false; } return 100 * (eStatus == CXFA_ContentLayoutProcessor::Result::kDone ? m_nProgressCounter : m_nProgressCounter - 1) / m_nProgressCounter; } bool CXFA_LayoutProcessor::IncrementLayout() { if (m_bNeedLayout) { RestartLayout(); return DoLayout() == 100; } return !m_bHasChangedContainers; } int32_t CXFA_LayoutProcessor::CountPages() const { return m_pViewLayoutProcessor ? m_pViewLayoutProcessor->GetPageCount() : 0; } CXFA_ViewLayoutItem* CXFA_LayoutProcessor::GetPage(int32_t index) const { return m_pViewLayoutProcessor ? m_pViewLayoutProcessor->GetPage(index) : nullptr; } CXFA_LayoutItem* CXFA_LayoutProcessor::GetLayoutItem(CXFA_Node* pFormItem) { return pFormItem->JSObject()->GetLayoutItem(); } void CXFA_LayoutProcessor::SetHasChangedContainer() { m_bHasChangedContainers = true; } bool CXFA_LayoutProcessor::NeedLayout() const { return m_bNeedLayout || m_bHasChangedContainers; }