/**************************************************************
 * 
 * Licensed to the Apache Software Foundation (ASF) under one
 * or more contributor license agreements.  See the NOTICE file
 * distributed with this work for additional information
 * regarding copyright ownership.  The ASF licenses this file
 * to you under the Apache License, Version 2.0 (the
 * "License"); you may not use this file except in compliance
 * with the License.  You may obtain a copy of the License at
 * 
 *   http://www.apache.org/licenses/LICENSE-2.0
 * 
 * Unless required by applicable law or agreed to in writing,
 * software distributed under the License is distributed on an
 * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
 * KIND, either express or implied.  See the License for the
 * specific language governing permissions and limitations
 * under the License.
 * 
 *************************************************************/



// MARKER(update_precomp.py): autogen include statement, do not remove
#include "precompiled_sw.hxx"

#include <editsh.hxx>
#include <doc.hxx>
#include <IDocumentUndoRedo.hxx>
#include <pam.hxx>
#include <docary.hxx>
#include <swundo.hxx>		// fuer die UndoIds
#include <section.hxx>
#include <edimp.hxx>
#include <sectfrm.hxx>		// SwSectionFrm
#include <cntfrm.hxx>		// SwCntntFrm
#include <tabfrm.hxx>		// SwTabFrm
#include <rootfrm.hxx>      // SwRootFrm


SwSection const*
SwEditShell::InsertSection(
        SwSectionData & rNewData, SfxItemSet const*const pAttr)
{
	const SwSection* pRet = 0;
	if( !IsTableMode() )
	{
		StartAllAction();
        GetDoc()->GetIDocumentUndoRedo().StartUndo( UNDO_INSSECTION, NULL );

		FOREACHPAM_START(this)
            SwSection const*const pNew =
                GetDoc()->InsertSwSection( *PCURCRSR, rNewData, 0, pAttr );
			if( !pRet )
				pRet = pNew;
		FOREACHPAM_END()

        GetDoc()->GetIDocumentUndoRedo().EndUndo( UNDO_INSSECTION, NULL );
		EndAllAction();
	}
	return pRet;
}


sal_Bool SwEditShell::IsInsRegionAvailable() const
{
	if( IsTableMode() )
		return sal_False;
	SwPaM* pCrsr = GetCrsr();
	if( pCrsr->GetNext() != pCrsr )
		return sal_False;
	if( pCrsr->HasMark() )
		return 0 != GetDoc()->IsInsRegionAvailable( *pCrsr );

	return sal_True;
}


const SwSection* SwEditShell::GetCurrSection() const
{
	if( IsTableMode() )
		return 0;

	return GetDoc()->GetCurrSection( *GetCrsr()->GetPoint() );
}

/*-----------------17.03.99 11:53-------------------
 * SwEditShell::GetAnySection liefert den fuer Spalten
 * zustaendigen Bereich, bei Fussnoten kann es nicht der
 * Bereich innerhalb der Fussnote sein.
 * --------------------------------------------------*/

const SwSection* SwEditShell::GetAnySection( sal_Bool bOutOfTab, const Point* pPt ) const
{
    SwFrm *pFrm;
	if ( pPt )
	{
		SwPosition aPos( *GetCrsr()->GetPoint() );
		Point aPt( *pPt );
		GetLayout()->GetCrsrOfst( &aPos, aPt );
		SwCntntNode *pNd = aPos.nNode.GetNode().GetCntntNode();
		pFrm = pNd->getLayoutFrm( GetLayout(), pPt );
	}
	else
		pFrm = GetCurrFrm( sal_False );

	if( bOutOfTab && pFrm )
		pFrm = pFrm->FindTabFrm();
	if( pFrm && pFrm->IsInSct() )
	{
		SwSectionFrm* pSect = pFrm->FindSctFrm();
		ASSERT( pSect, "GetAnySection: Where's my Sect?" );
		if( pSect->IsInFtn() && pSect->GetUpper()->IsInSct() )
		{
			pSect = pSect->GetUpper()->FindSctFrm();
			ASSERT( pSect, "GetAnySection: Where's my SectFrm?" );
		}
		return pSect->GetSection();
	}
	return NULL;
}

sal_uInt16 SwEditShell::GetSectionFmtCount() const
{
	return GetDoc()->GetSections().Count();
}


sal_Bool SwEditShell::IsAnySectionInDoc( sal_Bool bChkReadOnly, sal_Bool bChkHidden, sal_Bool bChkTOX ) const
{
	const SwSectionFmts& rFmts = GetDoc()->GetSections();
	sal_uInt16 nCnt = rFmts.Count();
	sal_uInt16 n;

	for( n = 0; n < nCnt; ++n )
	{
		SectionType eTmpType;
		const SwSectionFmt* pFmt = rFmts[ n ];
		if( pFmt->IsInNodesArr() &&
			(bChkTOX  ||
				( (eTmpType = pFmt->GetSection()->GetType()) != TOX_CONTENT_SECTION
				  && TOX_HEADER_SECTION != eTmpType ) ) )
		{
			const SwSection& rSect = *rFmts[ n ]->GetSection();
			if( (!bChkReadOnly && !bChkHidden ) ||
				(bChkReadOnly && rSect.IsProtectFlag() ) ||
				(bChkHidden && rSect.IsHiddenFlag() ) )
				break;
		}
	}
	return n != nCnt;
}

sal_uInt16 SwEditShell::GetSectionFmtPos( const SwSectionFmt& rFmt ) const
{
	SwSectionFmt* pFmt = (SwSectionFmt*)&rFmt;
	return GetDoc()->GetSections().GetPos( pFmt );
}

const SwSectionFmt& SwEditShell::GetSectionFmt( sal_uInt16 nFmt ) const
{
	return *GetDoc()->GetSections()[ nFmt ];
}


void SwEditShell::DelSectionFmt( sal_uInt16 nFmt )
{
	StartAllAction();
	GetDoc()->DelSectionFmt( GetDoc()->GetSections()[ nFmt ] );
	// rufe das AttrChangeNotify auf der UI-Seite.
	CallChgLnk();
	EndAllAction();
}


void SwEditShell::UpdateSection(sal_uInt16 const nSect,
        SwSectionData & rNewData, SfxItemSet const*const pAttr)
{
	StartAllAction();
    GetDoc()->UpdateSection( nSect, rNewData, pAttr );
	// rufe das AttrChangeNotify auf der UI-Seite.
	CallChgLnk();
	EndAllAction();
}

String SwEditShell::GetUniqueSectionName( const String* pChkStr ) const
{
	return GetDoc()->GetUniqueSectionName( pChkStr );
}

void SwEditShell::SetSectionAttr( const SfxItemSet& rSet,
									SwSectionFmt* pSectFmt )
{
	if( pSectFmt )
		_SetSectionAttr( *pSectFmt, rSet );
	else
	{
		// for all section in the selection

		FOREACHPAM_START(this)

			const SwPosition* pStt = PCURCRSR->Start(),
							* pEnd = PCURCRSR->End();

			const SwSectionNode* pSttSectNd = pStt->nNode.GetNode().FindSectionNode(),
							   * pEndSectNd = pEnd->nNode.GetNode().FindSectionNode();

			if( pSttSectNd || pEndSectNd )
			{
				if( pSttSectNd )
					_SetSectionAttr( *pSttSectNd->GetSection().GetFmt(),
									rSet );
				if( pEndSectNd && pSttSectNd != pEndSectNd )
					_SetSectionAttr( *pEndSectNd->GetSection().GetFmt(),
									rSet );

				if( pSttSectNd && pEndSectNd )
				{
					SwNodeIndex aSIdx( pStt->nNode );
					SwNodeIndex aEIdx( pEnd->nNode );
					if( pSttSectNd->EndOfSectionIndex() <
						pEndSectNd->GetIndex() )
					{
						aSIdx = pSttSectNd->EndOfSectionIndex() + 1;
						aEIdx = *pEndSectNd;
					}

					while( aSIdx < aEIdx )
					{
						if( 0 != (pSttSectNd = aSIdx.GetNode().GetSectionNode())
							|| ( aSIdx.GetNode().IsEndNode() &&
								0 != ( pSttSectNd = aSIdx.GetNode().
                                    StartOfSectionNode()->GetSectionNode())) )
							_SetSectionAttr( *pSttSectNd->GetSection().GetFmt(),
											rSet );
						aSIdx++;
					}
				}
			}

		FOREACHPAM_END()
	}
}

void SwEditShell::_SetSectionAttr( SwSectionFmt& rSectFmt,
									const SfxItemSet& rSet )
{
	StartAllAction();
	if(SFX_ITEM_SET == rSet.GetItemState(RES_CNTNT, sal_False))
	{
		SfxItemSet aSet(rSet);
		aSet.ClearItem(RES_CNTNT);
		GetDoc()->SetAttr( aSet, rSectFmt );
	}
	else
		GetDoc()->SetAttr( rSet, rSectFmt );

	// rufe das AttrChangeNotify auf der UI-Seite.
	CallChgLnk();
	EndAllAction();
}

// search inside the cursor selection for full selected sections.
// if any part of section in the selection return 0.
// if more than one in the selection return the count
sal_uInt16 SwEditShell::GetFullSelectedSectionCount() const
{
	sal_uInt16 nRet = 0;
	FOREACHPAM_START(this)

		const SwPosition* pStt = PCURCRSR->Start(),
						* pEnd = PCURCRSR->End();
		const SwCntntNode* pCNd;
		// check the selection, if Start at Node begin and End at Node end
		if( pStt->nContent.GetIndex() ||
			( 0 == ( pCNd = pEnd->nNode.GetNode().GetCntntNode() )) ||
			pCNd->Len() != pEnd->nContent.GetIndex() )
		{
			nRet = 0;
			break;
		}

// !!!!!!!!!!!!!!!!!!!!!!!!!!
// what about table at start or end ?
//		There is no selection possible!
// What about only a table inside the section ?
//		There is only a table selection possible!

		SwNodeIndex aSIdx( pStt->nNode, -1 ), aEIdx( pEnd->nNode, +1 );
		if( !aSIdx.GetNode().IsSectionNode() ||
			!aEIdx.GetNode().IsEndNode() ||
            !aEIdx.GetNode().StartOfSectionNode()->IsSectionNode() )
		{
			nRet = 0;
			break;
		}

		++nRet;
        if( &aSIdx.GetNode() != aEIdx.GetNode().StartOfSectionNode() )
			++nRet;

	FOREACHPAM_END()
	return nRet;
}


/**
 * Find the suitable node for a special insert (alt-enter).
 * This should enable inserting text before/after sections and tables.
 *
 * A node is found if:
 * 1) the innermost table/section is not in a write-protected area
 * 2) pCurrentPos is at or just before an end node
 *    (or at or just after a start node)
 * 3) there are only start/end nodes between pCurrentPos and the innermost
 *    table/section
 *
 * If a suitable node is found, an SwNode* is returned; else it is NULL.
 */
const SwNode* lcl_SpecialInsertNode(const SwPosition* pCurrentPos)
{
    const SwNode* pReturn = NULL;

    // the current position
    //    const SwPosition* pCurrentPos = GetCrsr()->GetPoint();
    DBG_ASSERT( pCurrentPos != NULL, "Strange, we have no position!" );
    const SwNode& rCurrentNode = pCurrentPos->nNode.GetNode();


    // find innermost section or table.  At the end of this scope,
    // pInntermostNode contain the section/table before/after which we should
    // insert our empty paragraph, or it will be NULL if none is found.
    const SwNode* pInnermostNode = NULL;
    {
        const SwNode* pTableNode = rCurrentNode.FindTableNode();
        const SwNode* pSectionNode = rCurrentNode.FindSectionNode();

        // find the table/section which is close
        if( pTableNode == NULL )
            pInnermostNode = pSectionNode;
        else if ( pSectionNode == NULL )
            pInnermostNode = pTableNode;
        else
        {
            // compare and choose the larger one
            pInnermostNode =
                ( pSectionNode->GetIndex() > pTableNode->GetIndex() )
                ? pSectionNode : pTableNode;
        }
    }

    // The previous version had a check to skip empty read-only sections. Those
    // shouldn't occur, so we only need to check whether our pInnermostNode is
    // inside a protected area.

    // Now, pInnermostNode is NULL or the innermost section or table node.
    if( (pInnermostNode != NULL) && !pInnermostNode->IsProtect() )
    {
        DBG_ASSERT( pInnermostNode->IsTableNode() ||
                    pInnermostNode->IsSectionNode(), "wrong node found" );
        DBG_ASSERT( ( pInnermostNode->GetIndex() <= rCurrentNode.GetIndex() )&&
                    ( pInnermostNode->EndOfSectionNode()->GetIndex() >=
                      rCurrentNode.GetIndex() ), "wrong node found" );

        // we now need to find the possible start/end positions

        // we found a start if
        // - we're at or just before a start node
        // - there are only start nodes between the current and pInnermostNode
        SwNodeIndex aBegin( pCurrentPos->nNode );
        if( rCurrentNode.IsCntntNode() &&
            (pCurrentPos->nContent.GetIndex() == 0))
            aBegin--;
        while( (aBegin != pInnermostNode->GetIndex()) &&
               aBegin.GetNode().IsStartNode() )
            aBegin--;
        bool bStart = ( aBegin == pInnermostNode->GetIndex() );

        // we found an end if
        // - we're at or just before an end node
        // - there are only end nodes between the current node and
        //   pInnermostNode's end node
        SwNodeIndex aEnd( pCurrentPos->nNode );
        if( rCurrentNode.IsCntntNode() &&
            ( pCurrentPos->nContent.GetIndex() ==
              rCurrentNode.GetCntntNode()->Len() ) )
            aEnd++;
        while( (aEnd != pInnermostNode->EndOfSectionNode()->GetIndex()) &&
               aEnd.GetNode().IsEndNode() )
            aEnd++;
        bool bEnd = ( aEnd == pInnermostNode->EndOfSectionNode()->GetIndex() );

        // evalutate result: if both start + end, end is preferred
        if( bEnd )
            pReturn = pInnermostNode->EndOfSectionNode();
        else if ( bStart )
            pReturn = pInnermostNode;
        // else pReturn = NULL;
    }
    // else: pReturn = NULL


    DBG_ASSERT( ( pReturn == NULL ) || pReturn->IsStartNode() ||
                                       pReturn->IsEndNode(),
                "SpecialInsertNode failed" );
    return pReturn;
}


/** a node can be special-inserted (alt-Enter) whenever lcl_SpecialInsertNode
    finds a suitable position
*/
bool SwEditShell::CanSpecialInsert() const
{
    return NULL != lcl_SpecialInsertNode( GetCrsr()->GetPoint() );
}


/** check whether a node cen be special-inserted (alt-Enter), and do so. Return
    whether insertion was possible.
 */
bool SwEditShell::DoSpecialInsert()
{
    bool bRet = false;

    // get current node
    SwPosition* pCursorPos = GetCrsr()->GetPoint();
    const SwNode* pInsertNode = lcl_SpecialInsertNode( pCursorPos );
    if( pInsertNode != NULL )
    {
		StartAllAction();

        // adjust insert position to insert before start nodes and after end
        // nodes
        SwNodeIndex aInsertIndex( *pInsertNode,
                                  pInsertNode->IsStartNode() ? -1 : 0 );
        SwPosition aInsertPos( aInsertIndex );

        // insert a new text node, and set the cursor
		bRet = GetDoc()->AppendTxtNode( aInsertPos );
		*pCursorPos = aInsertPos;

		// call AttrChangeNotify for the UI
		CallChgLnk();

		EndAllAction();
	}

	return bRet;
}

