31dbdd1766
Failure to invalidate sheet streams during reference update caused formula cross referenceing between sheets to totallly get borked. The bug was originally reported in i#116833.
452 lines
20 KiB
C++
452 lines
20 KiB
C++
/* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- */
|
|
/*************************************************************************
|
|
*
|
|
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
|
|
*
|
|
* Copyright 2000, 2010 Oracle and/or its affiliates.
|
|
*
|
|
* OpenOffice.org - a multi-platform office productivity suite
|
|
*
|
|
* This file is part of OpenOffice.org.
|
|
*
|
|
* OpenOffice.org is free software: you can redistribute it and/or modify
|
|
* it under the terms of the GNU Lesser General Public License version 3
|
|
* only, as published by the Free Software Foundation.
|
|
*
|
|
* OpenOffice.org is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
* GNU Lesser General Public License version 3 for more details
|
|
* (a copy is included in the LICENSE file that accompanied this code).
|
|
*
|
|
* You should have received a copy of the GNU Lesser General Public License
|
|
* version 3 along with OpenOffice.org. If not, see
|
|
* <http://www.openoffice.org/license.html>
|
|
* for a copy of the LGPLv3 License.
|
|
*
|
|
************************************************************************/
|
|
|
|
#ifndef SC_COLUMN_HXX
|
|
#define SC_COLUMN_HXX
|
|
|
|
#include "markarr.hxx"
|
|
#include "global.hxx"
|
|
#include "address.hxx"
|
|
#include "rangenam.hxx"
|
|
|
|
#include <set>
|
|
|
|
class Fraction;
|
|
class OutputDevice;
|
|
class Rectangle;
|
|
class SfxBroadcaster;
|
|
class SfxItemPoolCache;
|
|
class SfxItemSet;
|
|
class SvtListener;
|
|
class SfxPoolItem;
|
|
class SfxStyleSheetBase;
|
|
class SvxBorderLine;
|
|
class SvxBoxInfoItem;
|
|
class SvxBoxItem;
|
|
|
|
class ScAttrIterator;
|
|
class ScAttrArray;
|
|
struct ScAttrEntry;
|
|
class ScBaseCell;
|
|
class ScDocument;
|
|
class ScEditDataArray;
|
|
class ScFormulaCell;
|
|
class ScMarkData;
|
|
class ScPatternAttr;
|
|
class ScStyleSheet;
|
|
class SvtBroadcaster;
|
|
class TypedScStrCollection;
|
|
class ScProgress;
|
|
class ScPostIt;
|
|
struct ScFunctionData;
|
|
struct ScLineFlags;
|
|
struct ScMergePatternState;
|
|
class ScFlatBoolRowSegments;
|
|
struct ScSetStringParam;
|
|
struct ScColWidthParam;
|
|
|
|
#define COLUMN_DELTA 4
|
|
|
|
|
|
struct ScNeededSizeOptions
|
|
{
|
|
const ScPatternAttr* pPattern;
|
|
bool bFormula;
|
|
bool bSkipMerged;
|
|
bool bGetFont;
|
|
bool bTotalSize;
|
|
|
|
ScNeededSizeOptions()
|
|
{
|
|
pPattern = NULL;
|
|
bFormula = false;
|
|
bSkipMerged = true;
|
|
bGetFont = true;
|
|
bTotalSize = false;
|
|
}
|
|
};
|
|
|
|
struct ColEntry
|
|
{
|
|
SCROW nRow;
|
|
ScBaseCell* pCell;
|
|
};
|
|
|
|
|
|
class ScColumn
|
|
{
|
|
private:
|
|
SCCOL nCol;
|
|
SCTAB nTab;
|
|
|
|
SCSIZE nCount;
|
|
SCSIZE nLimit;
|
|
ColEntry* pItems;
|
|
|
|
ScAttrArray* pAttrArray;
|
|
ScDocument* pDocument;
|
|
|
|
friend class ScDocument; // for FillInfo
|
|
friend class ScDocumentIterator;
|
|
friend class ScValueIterator;
|
|
friend class ScDBQueryDataIterator;
|
|
friend class ScColumnIterator;
|
|
friend class ScQueryCellIterator;
|
|
friend class ScMarkedDataIter;
|
|
friend class ScCellIterator;
|
|
friend class ScHorizontalCellIterator;
|
|
friend class ScHorizontalAttrIterator;
|
|
|
|
public:
|
|
static bool bDoubleAlloc; // for Import: double size for alloc
|
|
|
|
class DoubleAllocSwitch
|
|
{
|
|
public:
|
|
DoubleAllocSwitch(bool bNewVal = true);
|
|
~DoubleAllocSwitch();
|
|
private:
|
|
bool mbOldVal;
|
|
};
|
|
public:
|
|
ScColumn();
|
|
~ScColumn();
|
|
|
|
void Init(SCCOL nNewCol, SCTAB nNewTab, ScDocument* pDoc);
|
|
|
|
bool Search( SCROW nRow, SCSIZE& nIndex ) const;
|
|
ScBaseCell* GetCell( SCROW nRow ) const;
|
|
void Insert( SCROW nRow, ScBaseCell* pCell );
|
|
void Insert( SCROW nRow, sal_uInt32 nFormatIndex, ScBaseCell* pCell );
|
|
void Append( SCROW nRow, ScBaseCell* pCell );
|
|
void Delete( SCROW nRow );
|
|
void DeleteAtIndex( SCSIZE nIndex );
|
|
void FreeAll();
|
|
void Resize( SCSIZE nSize );
|
|
void SwapRow( SCROW nRow1, SCROW nRow2 );
|
|
void SwapCell( SCROW nRow, ScColumn& rCol);
|
|
|
|
bool HasAttrib( SCROW nRow1, SCROW nRow2, sal_uInt16 nMask ) const;
|
|
bool HasAttribSelection( const ScMarkData& rMark, sal_uInt16 nMask ) const;
|
|
bool ExtendMerge( SCCOL nThisCol, SCROW nStartRow, SCROW nEndRow,
|
|
SCCOL& rPaintCol, SCROW& rPaintRow,
|
|
bool bRefresh, bool bAttrs );
|
|
|
|
bool IsEmptyVisData(bool bNotes) const; // without Broadcaster
|
|
bool IsEmptyData() const;
|
|
bool IsEmptyAttr() const;
|
|
bool IsEmpty() const;
|
|
|
|
// data only:
|
|
bool IsEmptyBlock(SCROW nStartRow, SCROW nEndRow, bool bIgnoreNotes = false) const;
|
|
SCSIZE GetEmptyLinesInBlock( SCROW nStartRow, SCROW nEndRow, ScDirection eDir ) const;
|
|
bool HasDataAt(SCROW nRow) const;
|
|
bool HasVisibleDataAt(SCROW nRow) const;
|
|
SCROW GetFirstDataPos() const;
|
|
SCROW GetLastDataPos() const;
|
|
SCROW GetLastVisDataPos(bool bNotes) const; // without Broadcaster
|
|
SCROW GetFirstVisDataPos(bool bNotes) const;
|
|
bool GetPrevDataPos(SCROW& rRow) const;
|
|
bool GetNextDataPos(SCROW& rRow) const;
|
|
void FindDataAreaPos(SCROW& rRow, long nMovY) const; // (without Broadcaster)
|
|
void FindUsed( SCROW nStartRow, SCROW nEndRow, bool* pUsed ) const;
|
|
|
|
SCSIZE VisibleCount( SCROW nStartRow, SCROW nEndRow ) const;
|
|
sal_uInt16 GetBlockMatrixEdges( SCROW nRow1, SCROW nRow2, sal_uInt16 nMask ) const;
|
|
bool HasSelectionMatrixFragment(const ScMarkData& rMark) const;
|
|
|
|
bool GetFirstVisibleAttr( SCROW& rFirstRow ) const;
|
|
bool GetLastVisibleAttr( SCROW& rLastRow ) const;
|
|
bool HasVisibleAttrIn( SCROW nStartRow, SCROW nEndRow ) const;
|
|
bool IsVisibleAttrEqual( const ScColumn& rCol, SCROW nStartRow = 0,
|
|
SCROW nEndRow = MAXROW ) const;
|
|
bool IsAllAttrEqual( const ScColumn& rCol, SCROW nStartRow, SCROW nEndRow ) const;
|
|
|
|
bool TestInsertCol( SCROW nStartRow, SCROW nEndRow) const;
|
|
bool TestInsertRow( SCSIZE nSize ) const;
|
|
void InsertRow( SCROW nStartRow, SCSIZE nSize );
|
|
void DeleteRow( SCROW nStartRow, SCSIZE nSize );
|
|
void DeleteRange( SCSIZE nStartIndex, SCSIZE nEndIndex, sal_uInt16 nDelFlag );
|
|
void DeleteArea(SCROW nStartRow, SCROW nEndRow, sal_uInt16 nDelFlag );
|
|
void CopyToClip(SCROW nRow1, SCROW nRow2, ScColumn& rColumn, bool bKeepScenarioFlags, bool bCloneNoteCaptions);
|
|
void CopyFromClip(SCROW nRow1, SCROW nRow2, long nDy,
|
|
sal_uInt16 nInsFlag, bool bAsLink, bool bSkipAttrForEmpty, ScColumn& rColumn);
|
|
void StartListeningInArea( SCROW nRow1, SCROW nRow2 );
|
|
void BroadcastInArea( SCROW nRow1, SCROW nRow2 );
|
|
|
|
void RemoveEditAttribs( SCROW nStartRow, SCROW nEndRow );
|
|
|
|
// Selection (?) of this document
|
|
void MixMarked( const ScMarkData& rMark, sal_uInt16 nFunction,
|
|
bool bSkipEmpty, ScColumn& rSrcCol );
|
|
void MixData( SCROW nRow1, SCROW nRow2, sal_uInt16 nFunction, bool bSkipEmpty,
|
|
ScColumn& rSrcCol );
|
|
|
|
ScFormulaCell* CreateRefCell( ScDocument* pDestDoc, const ScAddress& rDestPos,
|
|
SCSIZE nIndex, sal_uInt16 nFlags ) const;
|
|
|
|
ScAttrIterator* CreateAttrIterator( SCROW nStartRow, SCROW nEndRow ) const;
|
|
|
|
|
|
SCCOL GetCol() const { return nCol; }
|
|
|
|
// UpdateSelectionFunction: multi-select
|
|
void UpdateSelectionFunction( const ScMarkData& rMark,
|
|
ScFunctionData& rData,
|
|
ScFlatBoolRowSegments& rHiddenRows,
|
|
bool bDoExclude, SCROW nExStartRow, SCROW nExEndRow );
|
|
void UpdateAreaFunction( ScFunctionData& rData,
|
|
ScFlatBoolRowSegments& rHiddenRows,
|
|
SCROW nStartRow, SCROW nEndRow );
|
|
|
|
void CopyToColumn(SCROW nRow1, SCROW nRow2, sal_uInt16 nFlags, bool bMarked,
|
|
ScColumn& rColumn, const ScMarkData* pMarkData = NULL,
|
|
bool bAsLink = false );
|
|
void UndoToColumn(SCROW nRow1, SCROW nRow2, sal_uInt16 nFlags, bool bMarked,
|
|
ScColumn& rColumn, const ScMarkData* pMarkData = NULL );
|
|
|
|
void CopyScenarioFrom( const ScColumn& rSrcCol );
|
|
void CopyScenarioTo( ScColumn& rDestCol ) const;
|
|
bool TestCopyScenarioTo( const ScColumn& rDestCol ) const;
|
|
void MarkScenarioIn( ScMarkData& rDestMark ) const;
|
|
|
|
void CopyUpdated( const ScColumn& rPosCol, ScColumn& rDestCol ) const;
|
|
|
|
void SwapCol(ScColumn& rCol);
|
|
void MoveTo(SCROW nStartRow, SCROW nEndRow, ScColumn& rCol);
|
|
|
|
|
|
bool HasEditCells(SCROW nStartRow, SCROW nEndRow, SCROW& rFirst) const;
|
|
|
|
// TRUE = format for numbers is set
|
|
bool SetString( SCROW nRow, SCTAB nTab, const String& rString,
|
|
formula::FormulaGrammar::AddressConvention conv = formula::FormulaGrammar::CONV_OOO,
|
|
ScSetStringParam* pParam = NULL );
|
|
void SetValue( SCROW nRow, const double& rVal);
|
|
void SetError( SCROW nRow, const sal_uInt16 nError);
|
|
|
|
void GetString( SCROW nRow, String& rString ) const;
|
|
void GetInputString( SCROW nRow, String& rString ) const;
|
|
double GetValue( SCROW nRow ) const;
|
|
void GetFormula( SCROW nRow, String& rFormula ) const;
|
|
CellType GetCellType( SCROW nRow ) const;
|
|
SCSIZE GetCellCount() const { return nCount; }
|
|
sal_uInt32 GetWeightedCount() const;
|
|
sal_uInt32 GetCodeCount() const; // RPN-Code in formulas
|
|
sal_uInt16 GetErrCode( SCROW nRow ) const;
|
|
|
|
bool HasStringData( SCROW nRow ) const;
|
|
bool HasValueData( SCROW nRow ) const;
|
|
bool HasStringCells( SCROW nStartRow, SCROW nEndRow ) const;
|
|
|
|
/** Returns the pointer to a cell note object at the passed row. */
|
|
ScPostIt* GetNote( SCROW nRow );
|
|
/** Sets the passed cell note object at the passed row. Takes ownership! */
|
|
void TakeNote( SCROW nRow, ScPostIt* pNote );
|
|
/** Returns and forgets a cell note object at the passed row. */
|
|
ScPostIt* ReleaseNote( SCROW nRow );
|
|
/** Deletes the note at the passed row. */
|
|
void DeleteNote( SCROW nRow );
|
|
|
|
void SetDirty();
|
|
void SetDirty( const ScRange& );
|
|
void SetDirtyVar();
|
|
void SetDirtyAfterLoad();
|
|
void SetTableOpDirty( const ScRange& );
|
|
void CalcAll();
|
|
void CalcAfterLoad();
|
|
void CompileAll();
|
|
void CompileXML( ScProgress& rProgress );
|
|
|
|
void ResetChanged( SCROW nStartRow, SCROW nEndRow );
|
|
|
|
bool UpdateReference( UpdateRefMode eUpdateRefMode, SCCOL nCol1, SCROW nRow1, SCTAB nTab1,
|
|
SCCOL nCol2, SCROW nRow2, SCTAB nTab2,
|
|
SCsCOL nDx, SCsROW nDy, SCsTAB nDz,
|
|
ScDocument* pUndoDoc = NULL );
|
|
void UpdateInsertTab( SCTAB nTable);
|
|
void UpdateInsertTabOnlyCells( SCTAB nTable);
|
|
void UpdateDeleteTab( SCTAB nTable, bool bIsMove, ScColumn* pRefUndo = NULL );
|
|
void UpdateMoveTab(SCTAB nOldPos, SCTAB nNewPos, SCTAB nTabNo);
|
|
void UpdateCompile( bool bForceIfNameInUse = false );
|
|
void UpdateTranspose( const ScRange& rSource, const ScAddress& rDest,
|
|
ScDocument* pUndoDoc );
|
|
void UpdateGrow( const ScRange& rArea, SCCOL nGrowX, SCROW nGrowY );
|
|
|
|
void SetTabNo(SCTAB nNewTab);
|
|
bool IsRangeNameInUse(SCROW nRow1, SCROW nRow2, sal_uInt16 nIndex) const;
|
|
void FindRangeNamesInUse(SCROW nRow1, SCROW nRow2, std::set<sal_uInt16>& rIndexes) const;
|
|
void ReplaceRangeNamesInUse( SCROW nRow1, SCROW nRow2, const ScRangeData::IndexMap& rMap );
|
|
|
|
const SfxPoolItem* GetAttr( SCROW nRow, sal_uInt16 nWhich ) const;
|
|
const ScPatternAttr* GetPattern( SCROW nRow ) const;
|
|
const ScPatternAttr* GetMostUsedPattern( SCROW nStartRow, SCROW nEndRow ) const;
|
|
|
|
sal_uInt32 GetNumberFormat( SCROW nRow ) const;
|
|
sal_uInt32 GetNumberFormat( SCROW nStartRow, SCROW nEndRow ) const;
|
|
|
|
void MergeSelectionPattern( ScMergePatternState& rState, const ScMarkData& rMark, bool bDeep ) const;
|
|
void MergePatternArea( ScMergePatternState& rState, SCROW nRow1, SCROW nRow2, bool bDeep ) const;
|
|
void MergeBlockFrame( SvxBoxItem* pLineOuter, SvxBoxInfoItem* pLineInner,
|
|
ScLineFlags& rFlags,
|
|
SCROW nStartRow, SCROW nEndRow, bool bLeft, SCCOL nDistRight ) const;
|
|
void ApplyBlockFrame( const SvxBoxItem* pLineOuter, const SvxBoxInfoItem* pLineInner,
|
|
SCROW nStartRow, SCROW nEndRow, bool bLeft, SCCOL nDistRight );
|
|
|
|
void ApplyAttr( SCROW nRow, const SfxPoolItem& rAttr );
|
|
void ApplyPattern( SCROW nRow, const ScPatternAttr& rPatAttr );
|
|
void ApplyPatternArea( SCROW nStartRow, SCROW nEndRow, const ScPatternAttr& rPatAttr,
|
|
ScEditDataArray* pDataArray = NULL );
|
|
bool SetAttrEntries(ScAttrEntry* pData, SCSIZE nSize);
|
|
void SetPattern( SCROW nRow, const ScPatternAttr& rPatAttr, bool bPutToPool = false );
|
|
void SetPatternArea( SCROW nStartRow, SCROW nEndRow,
|
|
const ScPatternAttr& rPatAttr, bool bPutToPool = false );
|
|
void ApplyPatternIfNumberformatIncompatible( const ScRange& rRange,
|
|
const ScPatternAttr& rPattern, short nNewType );
|
|
|
|
void ApplyStyle( SCROW nRow, const ScStyleSheet& rStyle );
|
|
void ApplyStyleArea( SCROW nStartRow, SCROW nEndRow, const ScStyleSheet& rStyle );
|
|
void ApplySelectionStyle(const ScStyleSheet& rStyle, const ScMarkData& rMark);
|
|
void ApplySelectionLineStyle( const ScMarkData& rMark,
|
|
const SvxBorderLine* pLine, bool bColorOnly );
|
|
|
|
const ScStyleSheet* GetStyle( SCROW nRow ) const;
|
|
const ScStyleSheet* GetSelectionStyle( const ScMarkData& rMark, bool& rFound ) const;
|
|
const ScStyleSheet* GetAreaStyle( bool& rFound, SCROW nRow1, SCROW nRow2 ) const;
|
|
|
|
void FindStyleSheet( const SfxStyleSheetBase* pStyleSheet, ScFlatBoolRowSegments& rUsedRows, bool bReset );
|
|
bool IsStyleSheetUsed( const ScStyleSheet& rStyle, bool bGatherAllStyles ) const;
|
|
|
|
/// May return -1 if not found
|
|
SCsROW SearchStyle( SCsROW nRow, const ScStyleSheet* pSearchStyle,
|
|
bool bUp, bool bInSelection, const ScMarkData& rMark );
|
|
bool SearchStyleRange( SCsROW& rRow, SCsROW& rEndRow, const ScStyleSheet* pSearchStyle,
|
|
bool bUp, bool bInSelection, const ScMarkData& rMark );
|
|
|
|
bool ApplyFlags( SCROW nStartRow, SCROW nEndRow, sal_Int16 nFlags );
|
|
bool RemoveFlags( SCROW nStartRow, SCROW nEndRow, sal_Int16 nFlags );
|
|
void ClearItems( SCROW nStartRow, SCROW nEndRow, const sal_uInt16* pWhich );
|
|
|
|
void RemoveProtected( SCROW nStartRow, SCROW nEndRow );
|
|
|
|
SCsROW ApplySelectionCache( SfxItemPoolCache* pCache, const ScMarkData& rMark, ScEditDataArray* pDataArray = NULL );
|
|
void DeleteSelection( sal_uInt16 nDelFlag, const ScMarkData& rMark );
|
|
|
|
void ClearSelectionItems( const sal_uInt16* pWhich, const ScMarkData& rMark );
|
|
void ChangeSelectionIndent( bool bIncrement, const ScMarkData& rMark );
|
|
|
|
long GetNeededSize( SCROW nRow, OutputDevice* pDev,
|
|
double nPPTX, double nPPTY,
|
|
const Fraction& rZoomX, const Fraction& rZoomY,
|
|
bool bWidth, const ScNeededSizeOptions& rOptions );
|
|
sal_uInt16 GetOptimalColWidth( OutputDevice* pDev, double nPPTX, double nPPTY,
|
|
const Fraction& rZoomX, const Fraction& rZoomY,
|
|
bool bFormula, sal_uInt16 nOldWidth,
|
|
const ScMarkData* pMarkData,
|
|
const ScColWidthParam* pParam );
|
|
void GetOptimalHeight( SCROW nStartRow, SCROW nEndRow, sal_uInt16* pHeight,
|
|
OutputDevice* pDev,
|
|
double nPPTX, double nPPTY,
|
|
const Fraction& rZoomX, const Fraction& rZoomY,
|
|
bool bShrink, sal_uInt16 nMinHeight, SCROW nMinStart );
|
|
private:
|
|
long GetSimpleTextNeededSize( SCSIZE nIndex, OutputDevice* pDev,
|
|
bool bWidth );
|
|
public:
|
|
|
|
/// Including current, may return -1
|
|
SCsROW GetNextUnprotected( SCROW nRow, bool bUp ) const;
|
|
|
|
void GetFilterEntries(SCROW nStartRow, SCROW nEndRow, TypedScStrCollection& rStrings, bool& rHasDates);
|
|
bool GetDataEntries(SCROW nRow, TypedScStrCollection& rStrings, bool bLimit);
|
|
|
|
void UpdateInsertTabAbs(SCTAB nNewPos);
|
|
bool TestTabRefAbs(SCTAB nTable);
|
|
bool GetNextSpellingCell(SCROW& nRow, bool bInSel, const ScMarkData& rData) const;
|
|
|
|
void RemoveAutoSpellObj();
|
|
|
|
void StartListening( SvtListener& rLst, SCROW nRow );
|
|
void EndListening( SvtListener& rLst, SCROW nRow );
|
|
void MoveListeners( SvtBroadcaster& rSource, SCROW nDestRow );
|
|
void StartAllListeners();
|
|
void StartNeededListeners(); // only for cells where NeedsListening()==true
|
|
void SetRelNameDirty();
|
|
|
|
void CompileDBFormula();
|
|
void CompileDBFormula( bool bCreateFormulaString );
|
|
void CompileNameFormula( bool bCreateFormulaString );
|
|
void CompileColRowNameFormula();
|
|
|
|
sal_Int32 GetMaxStringLen( SCROW nRowStart, SCROW nRowEnd, CharSet eCharSet ) const;
|
|
xub_StrLen GetMaxNumberStringLen( sal_uInt16& nPrecision,
|
|
SCROW nRowStart, SCROW nRowEnd ) const;
|
|
|
|
private:
|
|
ScBaseCell* CloneCell(SCSIZE nIndex, sal_uInt16 nFlags, ScDocument& rDestDoc, const ScAddress& rDestPos);
|
|
};
|
|
|
|
|
|
class ScColumnIterator // walk through all data of a area/range
|
|
{
|
|
const ScColumn* pColumn;
|
|
SCSIZE nPos;
|
|
SCROW nTop;
|
|
SCROW nBottom;
|
|
public:
|
|
ScColumnIterator( const ScColumn* pCol, SCROW nStart=0, SCROW nEnd=MAXROW );
|
|
~ScColumnIterator();
|
|
|
|
bool Next( SCROW& rRow, ScBaseCell*& rpCell );
|
|
SCSIZE GetIndex() const;
|
|
};
|
|
|
|
|
|
class ScMarkedDataIter // walk through data in a selected area/range
|
|
{
|
|
const ScColumn* pColumn;
|
|
SCSIZE nPos;
|
|
ScMarkArrayIter* pMarkIter;
|
|
SCROW nTop;
|
|
SCROW nBottom;
|
|
bool bNext;
|
|
bool bAll;
|
|
|
|
public:
|
|
ScMarkedDataIter( const ScColumn* pCol, const ScMarkData* pMarkData,
|
|
bool bAllIfNone = false );
|
|
~ScMarkedDataIter();
|
|
|
|
bool Next( SCSIZE& rIndex );
|
|
};
|
|
|
|
|
|
#endif
|
|
|
|
|
|
/* vim:set shiftwidth=4 softtabstop=4 expandtab: */
|