{ fpspreadsheet.pas Writes an spreadsheet document AUTHORS: Felipe Monteiro de Carvalho } unit fpspreadsheet; {$ifdef fpc} {$mode delphi} {$endif} interface uses Classes, SysUtils, fpimage, AVL_Tree, avglvltree, lconvencoding; type {@@ File formats suppored by fpspreadsheet } TsSpreadsheetFormat = (sfExcel2, sfExcel3, sfExcel4, sfExcel5, sfExcel8, sfOOXML, sfOpenDocument, sfCSV, sfWikiTable_Pipes, sfWikiTable_WikiMedia); const { Default extensions } STR_EXCEL_EXTENSION = '.xls'; STR_OOXML_EXCEL_EXTENSION = '.xlsx'; STR_OPENDOCUMENT_CALC_EXTENSION = '.ods'; STR_COMMA_SEPARATED_EXTENSION = '.csv'; STR_WIKITABLE_PIPES = '.wikitable_pipes'; STR_WIKITABLE_WIKIMEDIA = '.wikitable_wikimedia'; type {@@ Possible encodings for a non-unicode encoded text } TsEncoding = ( seLatin1, seLatin2, seCyrillic, seGreek, seTurkish, seHebrew, seArabic ); {@@ Describes a formula Supported syntax:
    =A1+B1+C1/D2...  - Array with simple mathematical operations
    =SUM(A1:D1)      - SUM operation in a interval
    
} TsFormula = record FormulaStr: string; DoubleValue: double; end; {@@ Tokens to identify the elements in an expanded formula. See http://www.techonthenet.com/excel/formulas/ for an explanation of meaning and parameters of each formula NOTE: When adding or rearranging items: - make sure that the subtypes TOperandTokens, TBasicOperationTokens and TFuncTokens are complete - make sure to keep the FEProps table in sync - make sure to keep the TokenID table in TsSpreadBIFFWriter.FormulaElementKindToExcelTokenID, unit xlscommon, in sync } TFEKind = ( { Basic operands } fekCell, fekCellRef, fekCellRange, fekNum, fekInteger, fekString, fekBool, fekErr, fekMissingArg, { Basic operations } fekAdd, fekSub, fekMul, fekDiv, fekPercent, fekPower, fekUMinus, fekUPlus, fekConcat, // string concatenation fekEqual, fekGreater, fekGreaterEqual, fekLess, fekLessEqual, fekNotEqual, fekParen, { Built-in/Worksheet Functions} // math fekABS, fekACOS, fekACOSH, fekASIN, fekASINH, fekATAN, fekATANH, fekCOS, fekCOSH, fekDEGREES, fekEXP, fekINT, fekLN, fekLOG, fekLOG10, fekPI, fekRADIANS, fekRAND, fekROUND, fekSIGN, fekSIN, fekSINH, fekSQRT, fekTAN, fekTANH, // date/time fekDATE, fekDATEDIF, fekDATEVALUE, fekDAY, fekHOUR, fekMINUTE, fekMONTH, fekNOW, fekSECOND, fekTIME, fekTIMEVALUE, fekTODAY, fekWEEKDAY, fekYEAR, // statistical fekAVEDEV, fekAVERAGE, fekBETADIST, fekBETAINV, fekBINOMDIST, fekCHIDIST, fekCHIINV, fekCOUNT, fekCOUNTA, fekCOUNTBLANK, fekCOUNTIF, fekMAX, fekMEDIAN, fekMIN, fekPERMUT, fekPOISSON, fekPRODUCT, fekSTDEV, fekSTDEVP, fekSUM, fekSUMIF, fekSUMSQ, fekVAR, fekVARP, // financial fekFV, fekNPER, fekPMT, fekPV, fekRATE, // logical fekAND, fekFALSE, fekIF, fekNOT, fekOR, fekTRUE, // string fekCHAR, fekCODE, fekLEFT, fekLOWER, fekMID, fekPROPER, fekREPLACE, fekRIGHT, fekSUBSTITUTE, fekTRIM, fekUPPER, // lookup/reference fekCOLUMN, fekCOLUMNS, fekROW, fekROWS, // info fekCELLINFO, fekINFO, fekIsBLANK, fekIsERR, fekIsERROR, fekIsLOGICAL, fekIsNA, fekIsNONTEXT, fekIsNUMBER, fekIsRef, fekIsTEXT, fekValue, { Other operations } fekOpSUM {Unary sum operation. Note: CANNOT be used for summing sell contents; use fekSUM} ); {@@ These tokens identify operands in RPN formulas. } TOperandTokens = fekCell..fekMissingArg; {@@ These tokens identify basic operations in RPN formulas. } TBasicOperationTokens = fekAdd..fekParen; {@@ These tokens identify spreadsheet functions in RPN formulas. } TFuncTokens = fekAbs..fekOpSum; {@@ Flags to mark the address or a cell or a range of cells to be absolute or relative. They are used in the set TsRelFlags. } TsRelFlag = (rfRelRow, rfRelCol, rfRelRow2, rfRelCol2); {@@ Flags to mark the address of a cell or a range of cells to be absolute or relative. It is a set consisting of TsRelFlag elements. } TsRelFlags = set of TsRelFlag; {@@ Elements of an expanded formula. } TsFormulaElement = record ElementKind: TFEKind; Row, Row2: Word; // zero-based Col, Col2: Word; // zero-based Param1, Param2: Word; // Extra parameters DoubleValue: double; IntValue: Word; StringValue: String; RelFlags: TsRelFlags; // store info on relative/absolute addresses ParamsNum: Byte; end; {@@ Expanded formula. Used by backend modules. Provides more information than the text only. Consists of TsFormulaElements. } TsExpandedFormula = array of TsFormulaElement; {@@ RPN formula. Similar to the expanded formula, but in RPN notation. Simplifies the task of format writers which need RPN } TsRPNFormula = array of TsFormulaElement; {@@ Describes the type of content in a cell of a TsWorksheet } TCellContentType = (cctEmpty, cctFormula, cctRPNFormula, cctNumber, cctUTF8String, cctDateTime, cctBool, cctError); {@@ Error code values } TsErrorValue = ( errOK, // no error errEmptyIntersection, // #NULL! errDivideByZero, // #DIV/0! errWrongType, // #VALUE! errIllegalRef, // #REF! errWrongName, // #NAME? errOverflow, // #NUM! errArgError, // #N/A // --- no Excel errors -- errFormulaNotSupported ); {@@ List of possible formatting fields } TsUsedFormattingField = (uffTextRotation, uffFont, uffBold, uffBorder, uffBackgroundColor, uffNumberFormat, uffWordWrap, uffHorAlign, uffVertAlign ); {@@ Describes which formatting fields are active } TsUsedFormattingFields = set of TsUsedFormattingField; {@@ Number/cell formatting. Only uses a subset of the default formats, enough to be able to read/write date/time values. nfCustom allows to apply a format string directly. } TsNumberFormat = ( // general-purpose for all numbers nfGeneral, // numbers nfFixed, nfFixedTh, nfExp, nfSci, nfPercentage, // currency nfCurrency, nfCurrencyRed, nfAccounting, nfAccountingRed, // dates and times nfShortDateTime, {nfFmtDateTime, }nfShortDate, nfLongDate, nfShortTime, nfLongTime, nfShortTimeAM, nfLongTimeAM, nfTimeInterval, // other (format string goes directly into the file) nfCustom); const { @@ Codes for curreny format according to FormatSettings.CurrencyFormat: "C" = currency symbol, "V" = currency value, "S" = space character For the negative value formats, we use also: "B" = bracket, "M" = Minus The order of these characters represents the order of these items. Example: 1000 dollars --> "$1000" for pCV, or "1000 $" for pVsC -1000 dollars --> "($1000)" for nbCVb, or "-$ 1000" for nMCSV Assignment taken from "sysstr.inc" } pcfDefault = -1; // use value from Worksheet.FormatSettings.CurrencyFormat pcfCV = 0; // $1000 pcfVC = 1; // 1000$ pcfCSV = 2; // $ 1000 pcfVSC = 3; // 1000 $ ncfDefault = -1; // use value from Worksheet.FormatSettings.NegCurrFormat ncfBCVB = 0; // ($1000) ncfMCV = 1; // -$1000 ncfCMV = 2; // $-1000 ncfCVM = 3; // $1000- ncfBVCB = 4; // (1000$) ccfMVC = 5; // -1000$ ncfVMC = 6; // 1000-$ ncfVCM = 7; // 1000$- ncfMVSC = 8; // -1000 $ ncfMCSV = 9; // -$ 1000 ncfVSCM = 10; // 1000 $- ncfCSVM = 11; // $ 1000- ncfCSMV = 12; // $ -1000 ncfVMSC = 13; // 1000- $ ncfBCSVB = 14; // ($ 1000) ncfBVSCB = 15; // (1000 $) type {@@ Text rotation formatting. The text is rotated relative to the standard orientation, which is from left to right horizontal:
       --->
       ABC 
So 90 degrees clockwise means that the text will be:
       |  A
       |  B
       v  C 
And 90 degree counter clockwise will be:
       ^  C
       |  B
       |  A
Due to limitations of the text mode the characters are not rotated here. There is, however, also a "stacked" variant which looks exactly like the former case. } TsTextRotation = (trHorizontal, rt90DegreeClockwiseRotation, rt90DegreeCounterClockwiseRotation, rtStacked); {@@ Indicates horizontal text alignment in cells } TsHorAlignment = (haDefault, haLeft, haCenter, haRight); {@@ Indicates vertical text alignment in cells } TsVertAlignment = (vaDefault, vaTop, vaCenter, vaBottom); {@@ Colors in fpspreadsheet are given as indices into a palette. Use the workbook's GetPaletteColor to determine the color rgb value as little-endian (with "r" being the low-value byte, in agreement with TColor). The data type for rgb values is TsColorValue. } TsColor = Word; {@@ These are some constants for color indices into the default palette. Note, however, that if a different palette is used there may be more colors, and the names of the color constants may no longer be correct. } const scBlack = $00; scWhite = $01; scRed = $02; scGreen = $03; scBlue = $04; scYellow = $05; scMagenta = $06; scCyan = $07; scDarkRed = $08; scDarkGreen = $09; scDarkBlue = $0A; scNavy = $0A; scOlive = $0B; scPurple = $0C; scTeal = $0D; scSilver = $0E; scGrey = $0F; scGray = $0F; // redefine to allow different kinds of writing scGrey10pct = $10; scGray10pct = $10; scGrey20pct = $11; scGray20pct = $11; scOrange = $12; scDarkbrown = $13; scBrown = $14; scBeige = $15; scWheat = $16; // not sure - but I think the mechanism with scRGBColor is not working... // Will be removed sooner or later... scRGBColor = $FFFF; scNotDefined = $FFFF; type {@@ Data type for rgb color values } TsColorValue = DWord; {@@ Palette of color values. A "color value" is a DWord value containing rgb colors. } TsPalette = array[0..0] of TsColorValue; PsPalette = ^TsPalette; {@@ Font style (redefined to avoid usage of "Graphics" } TsFontStyle = (fssBold, fssItalic, fssStrikeOut, fssUnderline); {@@ Set of font styles } TsFontStyles = set of TsFontStyle; {@@ Font record used in fpspreadsheet. Contains the font name, the font size (in points), the font style, and the font color. } TsFont = class {@@ Name of the font face, such as 'Arial' or 'Times New Roman' } FontName: String; {@@ Size of the font in points } Size: Single; // in "points" {@@ Font style, such as bold, italics etc. - see TsFontStyle} Style: TsFontStyles; {@@ Text color given by the index into the workbook's color palette } Color: TsColor; end; {@@ Indicates the border for a cell. If included in the CellBorders set the corresponding border is drawn in the style defined by the CellBorderStyle. } TsCellBorder = (cbNorth, cbWest, cbEast, cbSouth); {@@ Indicates the border for a cell } TsCellBorders = set of TsCellBorder; {@@ Line style (for cell borders) } TsLineStyle = (lsThin, lsMedium, lsDashed, lsDotted, lsThick, lsDouble, lsHair); {@@ The Cell border style reocrd contains the linestyle and color of a cell border. There is a CellBorderStyle for each border. } TsCellBorderStyle = record LineStyle: TsLineStyle; Color: TsColor; end; {@@ The cell border styles of each cell border are collected in this array. } TsCellBorderStyles = array[TsCellBorder] of TsCellBorderStyle; {@@ Border styles for each cell border used by default: a thin, black, solid line } const DEFAULT_BORDERSTYLES: TsCellBorderStyles = ( (LineStyle: lsThin; Color: scBlack), (LineStyle: lsThin; Color: scBlack), (LineStyle: lsThin; Color: scBlack), (LineStyle: lsThin; Color: scBlack) ); type {@@ Cell structure for TsWorksheet The cell record contains information on the location of the cell (row and column index), on the value contained (number, date, text, ...), and on formatting. Never suppose that all *Value fields are valid, only one of the ContentTypes is valid. For other fields use TWorksheet.ReadAsUTF8Text and similar methods @see TWorksheet.ReadAsUTF8Text } TCell = record Col: Cardinal; // zero-based Row: Cardinal; // zero-based ContentType: TCellContentType; { Possible values for the cells } FormulaValue: TsFormula; RPNFormulaValue: TsRPNFormula; NumberValue: double; UTF8StringValue: ansistring; DateTimeValue: TDateTime; BoolValue: Boolean; ErrorValue: TsErrorValue; { Formatting fields } { When adding/deleting formatting fields don't forget to update CopyFormat! } UsedFormattingFields: TsUsedFormattingFields; FontIndex: Integer; TextRotation: TsTextRotation; HorAlignment: TsHorAlignment; VertAlignment: TsVertAlignment; Border: TsCellBorders; BorderStyles: TsCelLBorderStyles; BackgroundColor: TsColor; NumberFormat: TsNumberFormat; NumberFormatStr: String; RGBBackgroundColor: TFPColor; // only valid if BackgroundColor=scRGBCOLOR end; {@@ Pointer to a TCell record } PCell = ^TCell; const // Takes account of effect of cell margins on row height by adding this // value to the nominal row height. Note that this is an empirical value and may be wrong. ROW_HEIGHT_CORRECTION = 0.2; type {@@ The record TRow collects information of a spreadsheet row: @param Row The index of the row (beginning with 0) @param Height The height of the row (expressed as lines count of the default font) Only rows with heights that cannot be derived from the font height have a row record. } TRow = record Row: Cardinal; Height: Single; // in "lines" end; {@@ Pointer to a TRow record } PRow = ^TRow; {@@ The record TCol collects information on a spreadsheet column: @param Col The index of the column (beginning with 0) @param Width The width of the column (expressed in character count of the "0" character of the default font. Only columns with non-default widths have a column record. } TCol = record Col: Cardinal; Width: Single; // in "characters". Excel uses the width of char "0" in 1st font end; {@@ Pointer to a TCol record } PCol = ^TCol; {@@ User interface options: @param soShowGridLines Show or hide the grid lines in the spreadsheet @param soShowHeaders Show or hide the column or row headers of the spreadsheet @param soHasFrozenPanes If set a number of rows and columns of the spreadsheet is fixed and does not scroll. The number is defined by LeftPaneWidth and TopPaneHeight. @param soSelected (currently not used) } TsSheetOption = (soShowGridLines, soShowHeaders, soHasFrozenPanes, soSelected); {@@ Set of user interface options @ see TsSheetOption } TsSheetOptions = set of TsSheetOption; type TsCustomSpreadReader = class; TsCustomSpreadWriter = class; TsWorkbook = class; { TsWorksheet } {@@ This event fires whenever a cell value or cell formatting changes. It is handled by TsWorksheetGrid to update the grid. } TsCellEvent = procedure (Sender: TObject; ARow, ACol: Cardinal) of object; {@@ The worksheet contains a list of cells and provides a variety of methods to read or write data to the cells, or to change their formatting. } TsWorksheet = class private FWorkbook: TsWorkbook; FCells: TAvlTree; // Items are TCell FCurrentNode: TAVLTreeNode; // For GetFirstCell and GetNextCell FRows, FCols: TIndexedAVLTree; // This lists contain only rows or cols with styles different from default FLeftPaneWidth: Integer; FTopPaneHeight: Integer; FOptions: TsSheetOptions; FOnChangeCell: TsCellEvent; FOnChangeFont: TsCellEvent; function GetFormatSettings: TFormatSettings; procedure RemoveCallback(data, arg: pointer); protected procedure ChangedCell(ARow, ACol: Cardinal); procedure ChangedFont(ARow, ACol: Cardinal); public {@@ Name of the sheet. In the popular spreadsheet applications this is displayed at the tab of the sheet. } Name: string; { Base methods } constructor Create; destructor Destroy; override; { Utils } class function CellPosToText(ARow, ACol: Cardinal): string; procedure RemoveAllCells; { Reading of values } function ReadAsUTF8Text(ARow, ACol: Cardinal): ansistring; overload; function ReadAsUTF8Text(ACell: PCell): ansistring; overload; function ReadAsNumber(ARow, ACol: Cardinal): Double; overload; function ReadAsNumber(ACell: PCell): Double; overload; function ReadAsDateTime(ARow, ACol: Cardinal; out AResult: TDateTime): Boolean; overload; function ReadAsDateTime(ACell: PCell; out AResult: TDateTime): Boolean; overload; function ReadFormulaAsString(ACell: PCell): String; function ReadRPNFormulaAsString(ACell: PCell): String; function ReadUsedFormatting(ARow, ACol: Cardinal): TsUsedFormattingFields; function ReadBackgroundColor(ARow, ACol: Cardinal): TsColor; { Reading of cell attributes } function GetNumberFormatAttributes(ACell: PCell; out ADecimals: Byte; out ACurrencySymbol: String): Boolean; { Writing of values } procedure WriteBlank(ARow, ACol: Cardinal); procedure WriteBoolValue(ARow, ACol: Cardinal; AValue: Boolean); procedure WriteCurrency(ARow, ACol: Cardinal; AValue: Double; AFormat: TsNumberFormat = nfCurrency; ADecimals: Integer = 2; ACurrencySymbol: String = '?'; APosCurrFormat: Integer = -1; ANegCurrFormat: Integer = -1); overload; procedure WriteCurrency(ACell: PCell; AValue: Double; AFormat: TsNumberFormat = nfCurrency; ADecimals: Integer = -1; ACurrencySymbol: String = '?'; APosCurrFormat: Integer = -1; ANegCurrFormat: Integer = -1); overload; procedure WriteCurrency(ARow, ACol: Cardinal; AValue: Double; AFormat: TsNumberFormat; AFormatString: String); overload; procedure WriteCurrency(ACell: PCell; AValue: Double; AFormat: TsNumberFormat; AFormatString: String); overload; procedure WriteDateTime(ARow, ACol: Cardinal; AValue: TDateTime; AFormat: TsNumberFormat = nfShortDateTime; AFormatStr: String = ''); overload; procedure WriteDateTime(ACell: PCell; AValue: TDateTime; AFormat: TsNumberFormat = nfShortDateTime; AFormatStr: String = ''); overload; procedure WriteDateTime(ARow, ACol: Cardinal; AValue: TDateTime; AFormatStr: String); overload; procedure WriteDateTime(ACell: PCell; AValue: TDateTime; AFormatStr: String); overload; procedure WriteErrorValue(ARow, ACol: Cardinal; AValue: TsErrorValue); overload; procedure WriteErrorValue(ACell: PCell; AValue: TsErrorValue); overload; procedure WriteFormula(ARow, ACol: Cardinal; AFormula: TsFormula); procedure WriteNumber(ARow, ACol: Cardinal; ANumber: double; AFormat: TsNumberFormat = nfGeneral; ADecimals: Byte = 2); overload; procedure WriteNumber(ACell: PCell; ANumber: Double; AFormat: TsNumberFormat = nfGeneral; ADecimals: Byte = 2); overload; procedure WriteNumber(ARow, ACol: Cardinal; ANumber: double; AFormat: TsNumberFormat; AFormatString: String); overload; procedure WriteNumber(ACell: PCell; ANumber: Double; AFormat: TsNumberFormat; AFormatString: String); overload; procedure WriteRPNFormula(ARow, ACol: Cardinal; AFormula: TsRPNFormula); procedure WriteUTF8Text(ARow, ACol: Cardinal; AText: ansistring); overload; procedure WriteUTF8Text(ACell: PCell; AText: ansistring); overload; { Writing of cell attributes } procedure WriteBackgroundColor(ARow, ACol: Cardinal; AColor: TsColor); procedure WriteBorderColor(ARow, ACol: Cardinal; ABorder: TsCellBorder; AColor: TsColor); procedure WriteBorderLineStyle(ARow, ACol: Cardinal; ABorder: TsCellBorder; ALineStyle: TsLineStyle); procedure WriteBorders(ARow, ACol: Cardinal; ABorders: TsCellBorders); procedure WriteBorderStyle(ARow, ACol: Cardinal; ABorder: TsCellBorder; AStyle: TsCellBorderStyle); overload; procedure WriteBorderStyle(ARow, ACol: Cardinal; ABorder: TsCellBorder; ALineStyle: TsLineStyle; AColor: TsColor); overload; procedure WriteBorderStyles(ARow, ACol: Cardinal; const AStyles: TsCellBorderStyles); procedure WriteDecimals(ARow, ACol: Cardinal; ADecimals: byte); overload; procedure WriteDecimals(ACell: PCell; ADecimals: Byte); overload; function WriteFont(ARow, ACol: Cardinal; const AFontName: String; AFontSize: Single; AFontStyle: TsFontStyles; AFontColor: TsColor): Integer; overload; procedure WriteFont(ARow, ACol: Cardinal; AFontIndex: Integer); overload; function WriteFontColor(ARow, ACol: Cardinal; AFontColor: TsColor): Integer; function WriteFontName(ARow, ACol: Cardinal; AFontName: String): Integer; function WriteFontSize(ARow, ACol: Cardinal; ASize: Single): Integer; function WriteFontStyle(ARow, ACol: Cardinal; AStyle: TsFontStyles): Integer; procedure WriteHorAlignment(ARow, ACol: Cardinal; AValue: TsHorAlignment); procedure WriteNumberFormat(ARow, ACol: Cardinal; ANumberFormat: TsNumberFormat; const AFormatString: String = ''); overload; procedure WriteNumberFormat(ACell: PCell; ANumberFormat: TsNumberFormat; const AFormatString: String = ''); overload; procedure WriteTextRotation(ARow, ACol: Cardinal; ARotation: TsTextRotation); procedure WriteUsedFormatting(ARow, ACol: Cardinal; AUsedFormatting: TsUsedFormattingFields); procedure WriteVertAlignment(ARow, ACol: Cardinal; AValue: TsVertAlignment); procedure WriteWordwrap(ARow, ACol: Cardinal; AValue: boolean); { Data manipulation methods - For Cells } procedure CopyCell(AFromRow, AFromCol, AToRow, AToCol: Cardinal; AFromWorksheet: TsWorksheet); procedure CopyFormat(AFormat: PCell; AToRow, AToCol: Cardinal); overload; procedure CopyFormat(AFromCell, AToCell: PCell); overload; function FindCell(ARow, ACol: Cardinal): PCell; function GetCell(ARow, ACol: Cardinal): PCell; function GetCellCount: Cardinal; function GetFirstCell(): PCell; function GetNextCell(): PCell; function GetFirstCellOfRow(ARow: Cardinal): PCell; function GetLastCellOfRow(ARow: Cardinal): PCell; function GetLastColIndex: Cardinal; function GetLastColNumber: Cardinal; deprecated 'Use GetLastColIndex'; function GetLastRowIndex: Cardinal; function GetLastRowNumber: Cardinal; deprecated 'Use GetLastRowIndex'; { Data manipulation methods - For Rows and Cols } function CalcAutoRowHeight(ARow: Cardinal): Single; function FindRow(ARow: Cardinal): PRow; function FindCol(ACol: Cardinal): PCol; function GetCellCountInRow(ARow: Cardinal): Cardinal; function GetCellCountInCol(ACol: Cardinal): Cardinal; function GetRow(ARow: Cardinal): PRow; function GetRowHeight(ARow: Cardinal): Single; function GetCol(ACol: Cardinal): PCol; function GetColWidth(ACol: Cardinal): Single; procedure RemoveAllRows; procedure RemoveAllCols; procedure WriteRowInfo(ARow: Cardinal; AData: TRow); procedure WriteRowHeight(ARow: Cardinal; AHeight: Single); procedure WriteColInfo(ACol: Cardinal; AData: TCol); procedure WriteColWidth(ACol: Cardinal; AWidth: Single); { Properties } {@@ List of cells of the worksheet. Only cells with contents or with formatting are listed } property Cells: TAVLTree read FCells; {@@ List of all column records of the worksheet having a non-standard column width } property Cols: TIndexedAVLTree read FCols; {@@ FormatSettings for localization of some formatting strings } property FormatSettings: TFormatSettings read GetFormatSettings; {@@ List of all row records of the worksheet having a non-standard row height } property Rows: TIndexedAVLTree read FRows; {@@ Workbook to which the worksheet belongs } property Workbook: TsWorkbook read FWorkbook; // These are properties to interface to TsWorksheetGrid {@@ Parameters controlling visibility of grid lines and row/column headers, usage of frozen panes etc. } property Options: TsSheetOptions read FOptions write FOptions; {@@ Number of frozen columns which do not scroll } property LeftPaneWidth: Integer read FLeftPaneWidth write FLeftPaneWidth; {@@ Number of frozen rows which do not scroll } property TopPaneHeight: Integer read FTopPaneHeight write FTopPaneHeight; {@@ Event fired when cell contents or formatting changes } property OnChangeCell: TsCellEvent read FOnChangeCell write FOnChangeCell; {@@ Event fired when the font size in a cell changes } property OnChangeFont: TsCellEvent read FOnChangeFont write FOnChangeFont; end; { TsWorkbook } TsWorkbook = class private { Internal data } FWorksheets: TFPList; FEncoding: TsEncoding; FFormat: TsSpreadsheetFormat; FFontList: TFPList; FBuiltinFontCount: Integer; FPalette: array of TsColorValue; FReadFormulas: Boolean; FDefaultColWidth: Single; // in "characters". Excel uses the width of char "0" in 1st font FDefaultRowHeight: Single; // in "character heights", i.e. line count { Internal methods } procedure RemoveWorksheetsCallback(data, arg: pointer); public FormatSettings: TFormatSettings; { Base methods } constructor Create; destructor Destroy; override; class function GetFormatFromFileName(const AFileName: TFileName; out SheetType: TsSpreadsheetFormat): Boolean; function CreateSpreadReader(AFormat: TsSpreadsheetFormat): TsCustomSpreadReader; function CreateSpreadWriter(AFormat: TsSpreadsheetFormat): TsCustomSpreadWriter; procedure ReadFromFile(AFileName: string; AFormat: TsSpreadsheetFormat); overload; procedure ReadFromFile(AFileName: string); overload; procedure ReadFromFileIgnoringExtension(AFileName: string); procedure ReadFromStream(AStream: TStream; AFormat: TsSpreadsheetFormat); procedure WriteToFile(const AFileName: string; const AFormat: TsSpreadsheetFormat; const AOverwriteExisting: Boolean = False); overload; procedure WriteToFile(const AFileName: String; const AOverwriteExisting: Boolean = False); overload; procedure WriteToStream(AStream: TStream; AFormat: TsSpreadsheetFormat); { Worksheet list handling methods } function AddWorksheet(AName: string): TsWorksheet; function GetFirstWorksheet: TsWorksheet; function GetWorksheetByIndex(AIndex: Cardinal): TsWorksheet; function GetWorksheetByName(AName: String): TsWorksheet; function GetWorksheetCount: Cardinal; procedure RemoveAllWorksheets; { Font handling } function AddFont(const AFontName: String; ASize: Single; AStyle: TsFontStyles; AColor: TsColor): Integer; overload; function AddFont(const AFont: TsFont): Integer; overload; procedure CopyFontList(ASource: TFPList); function FindFont(const AFontName: String; ASize: Single; AStyle: TsFontStyles; AColor: TsColor): Integer; function GetDefaultFont: TsFont; function GetDefaultFontSize: Single; function GetFont(AIndex: Integer): TsFont; function GetFontCount: Integer; procedure InitFonts; procedure RemoveAllFonts; procedure SetDefaultFont(const AFontName: String; ASize: Single); { Color handling } function AddColorToPalette(AColorValue: TsColorValue): TsColor; function FPSColorToHexString(AColor: TsColor; ARGBColor: TFPColor): String; function GetColorName(AColorIndex: TsColor): string; function GetPaletteColor(AColorIndex: TsColor): TsColorValue; function GetPaletteColorAsHTMLStr(AColorIndex: TsColor): String; procedure SetPaletteColor(AColorIndex: TsColor; AColorValue: TsColorValue); function GetPaletteSize: Integer; procedure UseDefaultPalette; procedure UsePalette(APalette: PsPalette; APaletteCount: Word; ABigEndian: Boolean = false); {@@ The default column width given in "character units" (width of the character "0" in the default font) } property DefaultColWidth: Single read FDefaultColWidth; {@@ The default row height is given in "line count" (height of the default font } property DefaultRowHeight: Single read FDefaultRowHeight; {@@ This property is only used for formats which don't support unicode and support a single encoding for the whole document, like Excel 2 to 5 } property Encoding: TsEncoding read FEncoding write FEncoding; {@@ Identifies the file format which was detected when reading the file } property FileFormat: TsSpreadsheetFormat read FFormat; {@@ This property allows to turn off reading of rpn formulas; this is a precaution since formulas not correctly implemented by fpspreadsheet could crash the reading operation. } property ReadFormulas: Boolean read FReadFormulas write FReadFormulas; end; {@@ Contents of a number format record } TsNumFormatData = class public {@@ Excel refers to a number format by means of the format "index". } Index: Integer; {@@ OpenDocument refers to a number format by means of the format "name". } Name: String; {@@ Identifier of a built-in number format, see TsNumberFormat } NumFormat: TsNumberFormat; {@@ String of format codes, such as '#,##0.00', or 'hh:nn'. } FormatString: string; end; {@@ Specialized list for number format items } TsCustomNumFormatList = class(TFPList) private function GetItem(AIndex: Integer): TsNumFormatData; procedure SetItem(AIndex: Integer; AValue: TsNumFormatData); protected {@@ Workbook from which the number formats are collected in the list. It is mainly needed to get access to the FormatSettings for easy localization of some formatting strings. } FWorkbook: TsWorkbook; {@@ Identifies the first number format item that is written to the file. Items having a smaller index are not written. } FFirstFormatIndexInFile: Integer; {@@ Identifies the index of the next Excel number format item to be written. Needed for auto-creating of the user-defined Excel number format indexes } FNextFormatIndex: Integer; procedure AddBuiltinFormats; virtual; procedure RemoveFormat(AIndex: Integer); public constructor Create(AWorkbook: TsWorkbook); destructor Destroy; override; function AddFormat(AFormatCell: PCell): Integer; overload; function AddFormat(AFormatIndex: Integer; AFormatName, AFormatString: String; ANumFormat: TsNumberFormat): Integer; overload; function AddFormat(AFormatIndex: Integer; AFormatString: String; ANumFormat: TsNumberFormat): Integer; overload; function AddFormat(AFormatName, AFormatString: String; ANumFormat: TsNumberFormat): Integer; overload; function AddFormat(AFormatString: String; ANumFormat: TsNumberFormat): Integer; overload; procedure AnalyzeAndAdd(AFormatIndex: Integer; AFormatString: String); procedure Clear; procedure ConvertAfterReading(AFormatIndex: Integer; var AFormatString: String; var ANumFormat: TsNumberFormat); virtual; procedure ConvertBeforeWriting(var AFormatString: String; var ANumFormat: TsNumberFormat); virtual; procedure Delete(AIndex: Integer); function Find(ANumFormat: TsNumberFormat; AFormatString: String): Integer; overload; function Find(AFormatString: String): Integer; overload; function FindByIndex(AFormatIndex: Integer): Integer; function FindByName(AFormatName: String): Integer; function FindFormatOf(AFormatCell: PCell): integer; virtual; function FormatStringForWriting(AIndex: Integer): String; virtual; procedure Sort; {@@ Workbook from which the number formats are collected in the list. It is mainly needed to get access to the FormatSettings for easy localization of some formatting strings. } property Workbook: TsWorkbook read FWorkbook; {@@ Identifies the first number format item that is written to the file. Items having a smaller index are not written. } property FirstFormatIndexInFile: Integer read FFirstFormatIndexInFile; {@@ Number format items contained in the list } property Items[AIndex: Integer]: TsNumFormatData read GetItem write SetItem; default; end; {@@ TsSpreadReader class reference type } TsSpreadReaderClass = class of TsCustomSpreadReader; { TsCustomSpreadReader } TsCustomSpreadReader = class protected FWorkbook: TsWorkbook; FWorksheet: TsWorksheet; FNumFormatList: TsCustomNumFormatList; procedure CreateNumFormatList; virtual; { Record reading methods } procedure ReadBlank(AStream: TStream); virtual; abstract; procedure ReadFormula(AStream: TStream); virtual; abstract; procedure ReadLabel(AStream: TStream); virtual; abstract; procedure ReadNumber(AStream: TStream); virtual; abstract; public constructor Create(AWorkbook: TsWorkbook); virtual; // To allow descendents to override it destructor Destroy; override; { General writing methods } procedure ReadFromFile(AFileName: string; AData: TsWorkbook); virtual; procedure ReadFromStream(AStream: TStream; AData: TsWorkbook); virtual; procedure ReadFromStrings(AStrings: TStrings; AData: TsWorkbook); virtual; property Workbook: TsWorkbook read FWorkbook; property NumFormatList: TsCustomNumFormatList read FNumFormatList; end; {@@ TsSpreadWriter class reference type } TsSpreadWriterClass = class of TsCustomSpreadWriter; TCellsCallback = procedure (ACell: PCell; AStream: TStream) of object; { TsCustomSpreadWriter } TsCustomSpreadWriter = class private FWorkbook: TsWorkbook; protected FNumFormatList: TsCustomNumFormatList; { Helper routines } procedure AddDefaultFormats(); virtual; procedure CreateNumFormatList; virtual; function ExpandFormula(AFormula: TsFormula): TsExpandedFormula; function FindFormattingInList(AFormat: PCell): Integer; procedure FixFormat(ACell: PCell); virtual; procedure ListAllFormattingStylesCallback(ACell: PCell; AStream: TStream); procedure ListAllFormattingStyles; virtual; procedure ListAllNumFormatsCallback(ACell: PCell; AStream: TStream); procedure ListAllNumFormats; virtual; { Helpers for writing } procedure WriteCellCallback(ACell: PCell; AStream: TStream); procedure WriteCellsToStream(AStream: TStream; ACells: TAVLTree); { Record writing methods } procedure WriteBlank(AStream: TStream; const ARow, ACol: Cardinal; ACell: PCell); virtual; abstract; procedure WriteDateTime(AStream: TStream; const ARow, ACol: Cardinal; const AValue: TDateTime; ACell: PCell); virtual; abstract; procedure WriteFormula(AStream: TStream; const ARow, ACol: Cardinal; const AFormula: TsFormula; ACell: PCell); virtual; procedure WriteRPNFormula(AStream: TStream; const ARow, ACol: Cardinal; const AFormula: TsRPNFormula; ACell: PCell); virtual; procedure WriteLabel(AStream: TStream; const ARow, ACol: Cardinal; const AValue: string; ACell: PCell); virtual; abstract; procedure WriteNumber(AStream: TStream; const ARow, ACol: Cardinal; const AValue: double; ACell: PCell); virtual; abstract; public {@@ An array with cells which are models for the used styles In this array the Row property holds the Index to the corresponding XF field } FFormattingStyles: array of TCell; NextXFIndex: Integer; // Indicates which should be the next XF (Style) Index when filling the styles list constructor Create(AWorkbook: TsWorkbook); virtual; // To allow descendents to override it destructor Destroy; override; { General writing methods } procedure IterateThroughCells(AStream: TStream; ACells: TAVLTree; ACallback: TCellsCallback); procedure WriteToFile(const AFileName: string; const AOverwriteExisting: Boolean = False); virtual; procedure WriteToStream(AStream: TStream); virtual; procedure WriteToStrings(AStrings: TStrings); virtual; property Workbook: TsWorkbook read FWorkbook; property NumFormatList: TsCustomNumFormatList read FNumFormatList; end; {@@ List of registered formats } TsSpreadFormatData = record ReaderClass: TsSpreadReaderClass; WriterClass: TsSpreadWriterClass; Format: TsSpreadsheetFormat; end; {@@ Helper for simplification of RPN formula creation } PRPNItem = ^TRPNItem; TRPNItem = record FE: TsFormulaElement; Next: PRPNItem; end; {@@ Simple creation an RPNFormula array to be used in fpspreadsheet. For each formula element, use one of the RPNxxxx functions implemented here. They are designed to be nested into each other. Terminate the chain by using nil. Example: The RPN formula for the string expression "$A1+2" can be created as follows:
      var
        f: TsRPNFormula;

        f := CreateRPNFormula(
          RPNCellValue('A1',
          RPNNumber(2,
          RPNFunc(fekAdd,
          nil))));
    
} function CreateRPNFormula(AItem: PRPNItem): TsRPNFormula; procedure DestroyRPNFormula(AItem: PRPNItem); function RPNBool(AValue: Boolean; ANext: PRPNItem): PRPNItem; function RPNCellValue(ACellAddress: String; ANext: PRPNItem): PRPNItem; overload; function RPNCellValue(ARow, ACol: Integer; AFlags: TsRelFlags; ANext: PRPNItem): PRPNItem; overload; function RPNCellRef(ACellAddress: String; ANext: PRPNItem): PRPNItem; overload; function RPNCellRef(ARow, ACol: Integer; AFlags: TsRelFlags; ANext: PRPNItem): PRPNItem; overload; function RPNCellRange(ACellRangeAddress: String; ANext: PRPNItem): PRPNItem; overload; function RPNCellRange(ARow, ACol, ARow2, ACol2: Integer; AFlags: TsRelFlags; ANext: PRPNItem): PRPNItem; overload; function RPNErr(AErrCode: Byte; ANext: PRPNItem): PRPNItem; function RPNInteger(AValue: Word; ANext: PRPNItem): PRPNItem; function RPNMissingArg(ANext: PRPNItem): PRPNItem; function RPNNumber(AValue: Double; ANext: PRPNItem): PRPNItem; function RPNParenthesis(ANext: PRPNItem): PRPNItem; function RPNString(AValue: String; ANext: PRPNItem): PRPNItem; function RPNFunc(AToken: TFEKind; ANext: PRPNItem): PRPNItem; overload; function RPNFunc(AToken: TFEKind; ANumParams: Byte; ANext: PRPNItem): PRPNItem; overload; function FixedParamCount(AElementKind: TFEKind): Boolean; var GsSpreadFormats: array of TsSpreadFormatData; procedure RegisterSpreadFormat( AReaderClass: TsSpreadReaderClass; AWriterClass: TsSpreadWriterClass; AFormat: TsSpreadsheetFormat); procedure CopyCellFormat(AFromCell, AToCell: PCell); function GetFileFormatName(AFormat: TsSpreadsheetFormat): String; procedure MakeLEPalette(APalette: PsPalette; APaletteSize: Integer); implementation uses Math, StrUtils, TypInfo, fpsUtils, fpsNumFormatParser; { Translatable strings } resourcestring lpUnsupportedReadFormat = 'Tried to read a spreadsheet using an unsupported format'; lpUnsupportedWriteFormat = 'Tried to write a spreadsheet using an unsupported format'; lpNoValidSpreadsheetFile = '"%s" is not a valid spreadsheet file'; lpUnknownSpreadsheetFormat = 'unknown format'; lpInvalidFontIndex = 'Invalid font index'; lpInvalidNumberFormat = 'Trying to use an incompatible number format.'; lpInvalidDateTimeFormat = 'Trying to use an incompatible date/time format.'; lpNoValidNumberFormatString = 'No valid number format string.'; lpNoValidDateTimeFormatString = 'No valid date/time format string.'; lpNoValidCellAddress = '"%s" is not a valid cell address.'; lpNoValidCellRangeAddress = '"%s" is not a valid cell range address.'; lpIllegalNumberFormat = 'Illegal number format.'; lpSpecifyNumberOfParams = 'Specify number of parameters for function %s'; lpIncorrectParamCount = 'Funtion %s requires at least %d and at most %d parameters.'; lpTRUE = 'TRUE'; lpFALSE = 'FALSE'; lpErrEmptyIntersection = '#NULL!'; lpErrDivideByZero = '#DIV/0!'; lpErrWrongType = '#VALUE!'; lpErrIllegalRef = '#REF!'; lpErrWrongName = '#NAME?'; lpErrOverflow = '#NUM!'; lpErrArgError = '#N/A'; lpErrFormulaNotSupported = ''; var {@@ RGB colors RGB in "big-endian" notation (red at left). The values are inverted at initialization to be little-endian at run-time! The indices into this palette are named as scXXXX color constants. } DEFAULT_PALETTE: array[$00..$16] of TsColorValue = ( $000000, // $00: black $FFFFFF, // $01: white $FF0000, // $02: red $00FF00, // $03: green $0000FF, // $04: blue $FFFF00, // $05: yellow $FF00FF, // $06: magenta $00FFFF, // $07: cyan $800000, // $08: dark red $008000, // $09: dark green $000080, // $0A: dark blue $808000, // $0B: olive $800080, // $0C: purple $008080, // $0D: teal $C0C0C0, // $0E: silver $808080, // $0F: gray $E6E6E6, // $10: gray 10% $CCCCCC, // $11: gray 20% $FFA500, // $12: orange $A0522D, // $13: dark brown $CD853F, // $14: brown $F5F5DC, // $15: beige $F5DEB3 // $16: wheat ); {@@ Names of the colors of the DEFAULT_PALETTE } DEFAULT_COLORNAMES: array[$00..$16] of string = ( 'black', // 0 'white', // 1 'red', // 2 'green', // 3 'blue', // 4 'yellow', // 5 'magenta', // 6 'cyan', // 7 'dark red', // 8 'dark green', // 9 'dark blue', // $0A 'olive', // $0B 'purple', // $0C 'teal', // $0D 'silver', // $0E 'gray', // $0F 'gray 10%', // $10 'gray 20%', // $11 'orange', // $12 'dark brown', // $13 'brown', // $14 'beige', // $15 'wheat' // $16 ); { Properties of formula elements } type {@@ Properties of formula elements: @param Symbol Symbol used in the formula @param MinParams Minimum count of parameters used in this function @param MaxParams Maximum count of parameters used in this function } TFEProp = record Symbol: String; MinParams, MaxParams: Byte; end; const FEProps: array[TFEKind] of TFEProp = ( { Operands } (Symbol:''; MinParams:-1; MaxParams:Byte(-1)), // fekCell (Symbol:''; MinParams:-1; MaxParams:Byte(-1)), // fekCellRef (Symbol:''; MinParams:-1; MaxParams:Byte(-1)), // fekCellRange (Symbol:''; MinParams:-1; MaxParams:Byte(-1)), // fekCellNum (Symbol:''; MinParams:-1; MaxParams:Byte(-1)), // fekCellInteger (Symbol:''; MinParams:-1; MaxParams:Byte(-1)), // fekCellString (Symbol:''; MinParams:-1; MaxParams:Byte(-1)), // fekCellBool (Symbol:''; MinParams:-1; MaxParams:Byte(-1)), // fekCellErr (Symbol:''; MinParams:-1; MaxParams:Byte(-1)), // fekCellMissingArg { Basic operations } (Symbol:'+'; MinParams:2; MaxParams:2), // fekAdd (Symbol:'-'; MinParams:2; MaxParams:2), // fekSub (Symbol:'*'; MinParams:2; MaxParams:2), // fekMul (Symbol:'/'; MinParams:2; MaxParams:2), // fekDiv (Symbol:'%'; MinParams:1; MaxParams:1), // fekPercent (Symbol:'^'; MinParams:2; MaxParams:2), // fekPower (Symbol:'-'; MinParams:1; MaxParams:1), // fekUMinus (Symbol:'+'; MinParams:1; MaxParams:1), // fekUPlus (Symbol:'&'; MinParams:2; MaxParams:2), // fekConcat (string concatenation) (Symbol:'='; MinParams:2; MaxParams:2), // fekEqual (Symbol:'>'; MinParams:2; MaxParams:2), // fekGreater (Symbol:'>='; MinParams:2; MaxParams:2), // fekGreaterEqual (Symbol:'<'; MinParams:2; MaxParams:2), // fekLess (Symbol:'<='; MinParams:2; MaxParams:2), // fekLessEqual (Symbol:'<>'; MinParams:2; MaxParams:2), // fekNotEqual (Symbol:''; MinParams:1; MaxParams:1), // fekParen { math } (Symbol:'ABS'; MinParams:1; MaxParams:1), // fekABS (Symbol:'ACOS'; MinParams:1; MaxParams:1), // fekACOS (Symbol:'ACOSH'; MinParams:1; MaxParams:1), // fekACOSH (Symbol:'ASIN'; MinParams:1; MaxParams:1), // fekASIN (Symbol:'ASINH'; MinParams:1; MaxParams:1), // fekASINH (Symbol:'ATAN'; MinParams:1; MaxParams:1), // fekATAN (Symbol:'ATANH'; MinParams:1; MaxParams:1), // fekATANH, (Symbol:'COS'; MinParams:1; MaxParams:1), // fekCOS (Symbol:'COSH'; MinParams:1; MaxParams:1), // fekCOSH (Symbol:'DEGREES'; MinParams:1; MaxParams:1), // fekDEGREES (Symbol:'EXP'; MinParams:1; MaxParams:1), // fekEXP (Symbol:'INT'; MinParams:1; MaxParams:1), // fekINT (Symbol:'LN'; MinParams:1; MaxParams:1), // fekLN (Symbol:'LOG'; MinParams:1; MaxParams:2), // fekLOG, (Symbol:'LOG10'; MinParams:1; MaxParams:1), // fekLOG10 (Symbol:'PI'; MinParams:0; MaxParams:0), // fekPI (Symbol:'RADIANS'; MinParams:1; MaxParams:1), // fekRADIANS (Symbol:'RAND'; MinParams:0; MaxParams:0), // fekRAND (Symbol:'ROUND'; MinParams:2; MaxParams:2), // fekROUND, (Symbol:'SIGN'; MinParams:1; MaxParams:1), // fekSIGN (Symbol:'SIN'; MinParams:1; MaxParams:1), // fekSIN (Symbol:'SINH'; MinParams:1; MaxParams:1), // fekSINH (Symbol:'SQRT'; MinParams:1; MaxParams:1), // fekSQRT, (Symbol:'TAN'; MinParams:1; MaxParams:1), // fekTAN (Symbol:'TANH'; MinParams:1; MaxParams:1), // fekTANH, { date/time } (Symbol:'DATE'; MinParams:3; MaxParams:3), // fekDATE (Symbol:'DATEDIF'; MinParams:3; MaxParams:3), // fekDATEDIF (Symbol:'DATEVALUE'; MinParams:1; MaxParams:1), // fekDATEVALUE (Symbol:'DAY'; MinParams:1; MaxParams:1), // fekDAY (Symbol:'HOUR'; MinParams:1; MaxParams:1), // fekHOUR (Symbol:'MINUTE'; MinParams:1; MaxParams:1), // fekMINUTE (Symbol:'MONTH'; MinParams:1; MaxParams:1), // fekMONTH (Symbol:'NOW'; MinParams:0; MaxParams:0), // fekNOW (Symbol:'SECOND'; MinParams:1; MaxParams:1), // fekSECOND (Symbol:'TIME'; MinParams:3; MaxParams:3), // fekTIME (Symbol:'TIMEVALUE'; MinParams:1; MaxParams:1), // fekTIMEVALUE (Symbol:'TODAY'; MinParams:0; MaxParams:0), // fekTODAY (Symbol:'WEEKDAY'; MinParams:1; MaxParams:2), // fekWEEKDAY (Symbol:'YEAR'; MinParams:1; MaxParams:1), // fekYEAR { statistical } (Symbol:'AVEDEV'; MinParams:1; MaxParams:30), // fekAVEDEV (Symbol:'AVERAGE'; MinParams:1; MaxParams:30), // fekAVERAGE (Symbol:'BETADIST'; MinParams:3; MaxParams:5), // fekBETADIST (Symbol:'BETAINV'; MinParams:3; MaxParams:5), // fekBETAINV (Symbol:'BINOMDIST'; MinParams:4; MaxParams:4), // fekBINOMDIST (Symbol:'CHIDIST'; MinParams:2; MaxParams:2), // fekCHIDIST (Symbol:'CHIINV'; MinParams:2; MaxParams:2), // fekCHIINV (Symbol:'COUNT'; MinParams:0; MaxParams:30), // fekCOUNT (Symbol:'COUNTA'; MinParams:0; MaxParams:30), // fekCOUNTA (Symbol:'COUNTBLANK';MinParams:1; MaxParams:1), // fekCOUNTBLANK (Symbol:'COUNTIF'; MinParams:2; MaxParams:2), // fekCOUNTIF (Symbol:'MAX'; MinParams:1; MaxParams:30), // fekMAX (Symbol:'MEDIAN'; MinParams:1; MaxParams:30), // fekMEDIAN (Symbol:'MIN'; MinParams:1; MaxParams:30), // fekMIN (Symbol:'PERMUT'; MinParams:2; MaxParams:2), // fekPERMUT (Symbol:'POISSON'; MinParams:3; MaxParams:3), // fekPOISSON (Symbol:'PRODUCT'; MinParams:0; MaxParams:30), // fekPRODUCT (Symbol:'STDEV'; MinParams:1; MaxParams:30), // fekSTDEV (Symbol:'STDEVP'; MinParams:1; MaxParams:30), // fekSTDEVP (Symbol:'SUM'; MinParams:0; MaxParams:30), // fekSUM (Symbol:'SUMIF'; MinParams:2; MaxParams:3), // fekSUMIF (Symbol:'SUMSQ'; MinParams:0; MaxParams:30), // fekSUMSQ (Symbol:'VAR'; MinParams:1; MaxParams:30), // fekVAR (Symbol:'VARP'; MinParams:1; MaxParams:30), // fekVARP { financial } (Symbol:'FV'; MinParams:3; MaxParams:5), // fekFV (Symbol:'NPER'; MinParams:3; MaxParams:5), // fekNPER (Symbol:'PMT'; MinParams:3; MaxParams:5), // fekPMT (Symbol:'PV'; MinParams:3; MaxParams:5), // fekPV (Symbol:'RATE'; MinParams:3; MaxParams:6), // fekRATE { logical } (Symbol:'AND'; MinParams:0; MaxParams:30), // fekAND (Symbol:'FALSE'; MinParams:0; MaxParams:0), // fekFALSE (Symbol:'IF'; MinParams:2; MaxParams:3), // fekIF (Symbol:'NOT'; MinParams:1; MaxParams:1), // fekNOT (Symbol:'OR'; MinParams:1; MaxParams:30), // fekOR (Symbol:'TRUE'; MinParams:0; MaxParams:0), // fekTRUE { string } (Symbol:'CHAR'; MinParams:1; MaxParams:1), // fekCHAR (Symbol:'CODE'; MinParams:1; MaxParams:1), // fekCODE (Symbol:'LEFT'; MinParams:1; MaxParams:2), // fekLEFT (Symbol:'LOWER'; MinParams:1; MaxParams:1), // fekLOWER (Symbol:'MID'; MinParams:3; MaxParams:3), // fekMID (Symbol:'PROPER'; MinParams:1; MaxParams:1), // fekPROPER (Symbol:'REPLACE'; MinParams:4; MaxParams:4), // fekREPLACE (Symbol:'RIGHT'; MinParams:1; MaxParams:2), // fekRIGHT (Symbol:'SUBSTITUTE';MinParams:3; MaxParams:4), // fekSUBSTITUTE (Symbol:'TRIM'; MinParams:1; MaxParams:1), // fekTRIM (Symbol:'UPPER'; MinParams:1; MaxParams:1), // fekUPPER { lookup/reference } (Symbol:'COLUMN'; MinParams:0; MaxParams:1), // fekCOLUMN (Symbol:'COLUMNS'; MinParams:1; MaxParams:1), // fekCOLUMNS (Symbol:'ROW'; MinParams:0; MaxParams:1), // fekROW (Symbol:'ROWS'; MinParams:1; MaxParams:1), // fekROWS { info } (Symbol:'CELL'; MinParams:1; MaxParams:2), // fekCELLINFO (Symbol:'INFO'; MinParams:1; MaxParams:1), // fekINFO (Symbol:'ISBLANK'; MinParams:1; MaxParams:1), // fekIsBLANK (Symbol:'ISERR'; MinParams:1; MaxParams:1), // fekIsERR (Symbol:'ISERROR'; MinParams:1; MaxParams:1), // fekIsERROR (Symbol:'ISLOGICAL'; MinParams:1; MaxParams:1), // fekIsLOGICAL (Symbol:'ISNA'; MinParams:1; MaxParams:1), // fekIsNA (Symbol:'ISNONTEXT'; MinParams:1; MaxParams:1), // fekIsNONTEXT (Symbol:'ISNUMBER'; MinParams:1; MaxParams:1), // fekIsNUMBER (Symbol:'ISREF'; MinParams:1; MaxParams:1), // fekIsRef (Symbol:'ISTEXT'; MinParams:1; MaxParams:1), // fekIsTEXT (Symbol:'VALUE'; MinParams:1; MaxParams:1), // fekValue { Other operations } (Symbol:'SUM'; MinParams:1; MaxParams:1) // fekOpSUM (Unary sum operation). Note: CANNOT be used for summing sell contents; use fekSUM} ); {@@ Registers a new reader/writer pair for a format } procedure RegisterSpreadFormat( AReaderClass: TsSpreadReaderClass; AWriterClass: TsSpreadWriterClass; AFormat: TsSpreadsheetFormat); var len: Integer; begin len := Length(GsSpreadFormats); SetLength(GsSpreadFormats, len + 1); GsSpreadFormats[len].ReaderClass := AReaderClass; GsSpreadFormats[len].WriterClass := AWriterClass; GsSpreadFormats[len].Format := AFormat; end; {@@ Returns the name of the given file format. @param AFormat Identifier of the file format @return 'BIFF2', 'BIFF3', 'BIFF4', 'BIFF5', 'BIFF8', 'OOXML', 'Open Document', 'CSV, 'WikiTable Pipes', or 'WikiTable WikiMedia" } function GetFileFormatName(AFormat: TsSpreadsheetFormat): string; begin case AFormat of sfExcel2 : Result := 'BIFF2'; sfExcel3 : Result := 'BIFF3'; sfExcel4 : Result := 'BIFF4'; sfExcel5 : Result := 'BIFF5'; sfExcel8 : Result := 'BIFF8'; sfooxml : Result := 'OOXML'; sfOpenDocument : Result := 'Open Document'; sfCSV : Result := 'CSV'; sfWikiTable_Pipes : Result := 'WikiTable Pipes'; sfWikiTable_WikiMedia : Result := 'WikiTable WikiMedia'; else Result := lpUnknownSpreadsheetFormat; end; end; {@@ If a palette is coded as big-endian (e.g. by copying the rgb values from the OpenOffice doc) the palette values can be converted by means of this procedure to little-endian which is required internally by TsWorkbook. @param APalette Pointer to the palette to be converted. After conversion, its color values are replaced. @param APaletteSize Number of colors contained in the palette } procedure MakeLEPalette(APalette: PsPalette; APaletteSize: Integer); var i: Integer; begin for i := 0 to APaletteSize-1 do {$IFDEF RNGCHECK} {$R-} {$ENDIF} APalette^[i] := LongRGBToExcelPhysical(APalette^[i]) {$IFDEF RNGCHECK} {$R+} {$ENDIF} end; {@@ Copies the format of a cell to another one. @param AFromCell cell from which the format is to be copied @param AToCell cell to which the format is to be copied } procedure CopyCellFormat(AFromCell, AToCell: PCell); begin Assert(AFromCell <> nil); Assert(AToCell <> nil); AToCell^.UsedFormattingFields := AFromCell^.UsedFormattingFields; AToCell^.BackgroundColor := AFromCell^.BackgroundColor; AToCell^.Border := AFromCell^.Border; AToCell^.BorderStyles := AFromCell^.BorderStyles; AToCell^.FontIndex := AFromCell^.FontIndex; AToCell^.HorAlignment := AFromCell^.HorAlignment; AToCell^.VertAlignment := AFromCell^.VertAlignment; AToCell^.TextRotation := AFromCell^.TextRotation; AToCell^.NumberFormat := AFromCell^.NumberFormat; AToCell^.NumberFormatStr := AFromCell^.NumberFormatStr; end; { TsWorksheet } {@@ Helper method for clearing the records in a spreadsheet. } procedure TsWorksheet.RemoveCallback(data, arg: pointer); begin Unused(arg); { The strings and dyn arrays must be reset to nil content manually, because FreeMem only frees the record mem, without checking its content } PCell(data).UTF8StringValue := ''; PCell(data).NumberFormatStr := ''; SetLength(PCell(data).RPNFormulaValue, 0); FreeMem(data); end; function CompareCells(Item1, Item2: Pointer): Integer; begin result := PCell(Item1).Row - PCell(Item2).Row; if Result = 0 then Result := PCell(Item1).Col - PCell(Item2).Col; end; function CompareRows(Item1, Item2: Pointer): Integer; begin result := PRow(Item1).Row - PRow(Item2).Row; end; function CompareCols(Item1, Item2: Pointer): Integer; begin result := PCol(Item1).Col - PCol(Item2).Col; end; {@@ Constructor of the TsWorksheet class. } constructor TsWorksheet.Create; begin inherited Create; FCells := TAVLTree.Create(@CompareCells); FRows := TIndexedAVLTree.Create(@CompareRows); FCols := TIndexedAVLTree.Create(@CompareCols); FOptions := [soShowGridLines, soShowHeaders]; end; {@@ Destructor of the TsWorksheet class. } destructor TsWorksheet.Destroy; begin RemoveAllCells; RemoveAllRows; RemoveAllCols; FCells.Free; FRows.Free; FCols.Free; inherited Destroy; end; {@@ Converts a FPSpreadsheet cell position, which is Row, Col in numbers and zero based, to a textual representation which is [Col][Row], being that the Col is in letters and the row is in 1-based numbers } class function TsWorksheet.CellPosToText(ARow, ACol: Cardinal): string; var lStr: string; begin lStr := ''; if ACol < 26 then lStr := Char(ACol+65); Result := Format('%s%d', [lStr, ARow+1]); end; {@@ Is called whenever a cell value or formatting has changed. Fires an event "OnChangeCell". This is handled by TsWorksheetGrid to update the grid cell. @param ARow Row index of the cell which has been changed @param ACol Column index of the cell which has been changed } procedure TsWorksheet.ChangedCell(ARow, ACol: Cardinal); begin if Assigned(FOnChangeCell) then FOnChangeCell(Self, ARow, ACol); end; {@@ Is called whenever a font height changes. Fires an even "OnChangeFont" which is handled by TsWorksheetGrid to update the row heights. @param ARow Row index of the cell for which the font height has changed @param ACol Column index of the cell for which the font height has changed. } procedure TsWorksheet.ChangedFont(ARow, ACol: Cardinal); begin if Assigned(FonChangeFont) then FOnChangeFont(Self, ARow, ACol); end; {@@ Copies a cell. The source cell can be located in a different worksheet, while the destination cell must be in the same worksheet which calls the methode. @param AFromRow Row index of the source cell @param AFromCol Column index of the source cell @param AToRow Row index of the destination cell @param AToCol Column index of the destination cell @param AFromWorksheet Worksheet containing the source cell. } procedure TsWorksheet.CopyCell(AFromRow, AFromCol, AToRow, AToCol: Cardinal; AFromWorksheet: TsWorksheet); var lSrcCell, lDestCell: PCell; begin lSrcCell := AFromWorksheet.FindCell(AFromRow, AFromCol); lDestCell := GetCell(AToRow, AToCol); lDestCell^ := lSrcCell^; lDestCell^.Row := AToRow; lDestCell^.Col := AToCol; ChangedCell(AToRow, AToCol); ChangedFont(AToRow, AToCol); { lCurStr := AFromWorksheet.ReadAsUTF8Text(AFromRow, AFromCol); lCurUsedFormatting := AFromWorksheet.ReadUsedFormatting(AFromRow, AFromCol); lCurColor := AFromWorksheet.ReadBackgroundColor(AFromRow, AFromCol); WriteUTF8Text(AToRow, AToCol, lCurStr); WriteUsedFormatting(AToRow, AToCol, lCurUsedFormatting); if uffBackgroundColor in lCurUsedFormatting then begin WriteBackgroundColor(AToRow, AToCol, lCurColor); end; } end; {@@ Copies all format parameters from the format cell to another cell. @param AFromCell Pointer to source cell @param AToCell Pointer to destination cell } procedure TsWorksheet.CopyFormat(AFromCell, AToCell: PCell); begin if (AFromCell = nil) or (AToCell = nil) then exit; CopyCellFormat(AFromCell, AToCell); ChangedCell(AToCell^.Row, AToCell^.Col); ChangedFont(AToCell^.Row, AToCell^.Col); end; {@@ Copies all format parameters from a given cell to another cell identified by its row/column indexes. @param AFormat Pointer to the source cell from which the format is copied. @param AToRow Row index of the destination cell @param AToCol Column index of the destination cell } procedure TsWorksheet.CopyFormat(AFormat: PCell; AToRow, AToCol: Cardinal); begin CopyFormat(AFormat, GetCell(AToRow, AToCol)); end; {@@ Tries to locate a Cell in the list of already written Cells @param ARow The row of the cell @param ACol The column of the cell @return Pointer to the cell if found, or nil if not found @see TCell } function TsWorksheet.FindCell(ARow, ACol: Cardinal): PCell; var LCell: TCell; AVLNode: TAVLTreeNode; begin Result := nil; LCell.Row := ARow; LCell.Col := ACol; AVLNode := FCells.Find(@LCell); if Assigned(AVLNode) then result := PCell(AVLNode.Data); end; {@@ Obtains an allocated cell at the desired location. If the Cell already exists, a pointer to it will be returned. If not, then new memory for the cell will be allocated, a pointer to it will be returned and it will be added to the list of cells. @param ARow Row index of the cell @param ACol Column index of the cell @return A pointer to the Cell on the desired location. @see TCell } function TsWorksheet.GetCell(ARow, ACol: Cardinal): PCell; begin Result := FindCell(ARow, ACol); if (Result = nil) then begin Result := GetMem(SizeOf(TCell)); FillChar(Result^, SizeOf(TCell), #0); Result^.Row := ARow; Result^.Col := ACol; Result^.ContentType := cctEmpty; Result^.BorderStyles := DEFAULT_BORDERSTYLES; Cells.Add(Result); end; end; {@@ Returns the number of cells in the worksheet with contents. This routine is used together with GetFirstCell and GetNextCell to iterate througth all cells in a worksheet efficiently. @return The number of cells with contents in the worksheet @see TCell @see GetFirstCell @see GetNextCell } function TsWorksheet.GetCellCount: Cardinal; begin Result := FCells.Count; end; {@@ Determines some number format attributes (decimal places, currency symbol) of a cell @param ACell Pointer to the cell under investigation @param ADecimals Number of decimal places that can be extracted from the formatting string, e.g. in case of '0.000' this would be 3. @param ACurrencySymbol String representing the currency symbol extracted from the formatting string. @return true if the the format string could be analyzed successfully, false if not } function TsWorksheet.GetNumberFormatAttributes(ACell: PCell; out ADecimals: byte; out ACurrencySymbol: String): Boolean; var parser: TsNumFormatParser; nf: TsNumberFormat; begin Result := false; if ACell <> nil then begin parser := TsNumFormatParser.Create(FWorkbook, ACell^.NumberFormatStr); try if parser.Status = psOK then begin nf := parser.NumFormat; if (nf = nfGeneral) or IsDateTimeFormat(nf) then begin ADecimals := 2; ACurrencySymbol := '?'; end else begin ADecimals := parser.Decimals; ACurrencySymbol := parser.CurrencySymbol; end; Result := true; end; finally parser.Free; end; end; end; {@@ Returns the first Cell. Use together with GetCellCount and GetNextCell to iterate througth all cells in a worksheet efficiently. @return The first cell if any exists, nil otherwise @see TCell @see GetCellCount @see GetNextCell } function TsWorksheet.GetFirstCell(): PCell; begin FCurrentNode := FCells.FindLowest(); if FCurrentNode <> nil then Result := PCell(FCurrentNode.Data) else Result := nil; end; {@@ Returns the next Cell. Should always be used either after GetFirstCell or after GetNextCell. Use together with GetCellCount and GetFirstCell to iterate througth all cells in a worksheet efficiently. @return The first cell if any exists, nil otherwise @see TCell @see GetCellCount @see GetFirstCell } function TsWorksheet.GetNextCell(): PCell; begin FCurrentNode := FCells.FindSuccessor(FCurrentNode); if FCurrentNode <> nil then Result := PCell(FCurrentNode.Data) else Result := nil; end; {@@ Returns the 0-based index of the last column with a cell with contents. If no cells have contents, zero will be returned, which is also a valid value. Use GetCellCount to verify if there is at least one cell with contents in the worksheet. @see GetCellCount } function TsWorksheet.GetLastColIndex: Cardinal; var AVLNode: TAVLTreeNode; i: Integer; begin Result := 0; // Traverse the tree from lowest to highest. // Since tree primary sort order is on Row // highest Col could exist anywhere. AVLNode := FCells.FindLowest; While Assigned(AVLNode) do begin Result := Math.Max(Result, PCell(AVLNode.Data)^.Col); AVLNode := FCells.FindSuccessor(AVLNode); end; // In addition, there may be column records defining the column width even // without content for i:=0 to FCols.Count-1 do if FCols[i] <> nil then Result := Math.Max(Result, PCol(FCols[i])^.Col); end; {@@ Deprecated, use GetLastColIndex instead @see GetLastColIndex } function TsWorksheet.GetLastColNumber: Cardinal; begin Result := GetLastColIndex; end; {@@ Finds the first cell with contents in a given row @param ARow Index of the row considered @return Pointer to the first cell in this row, or nil if the row is empty. } function TsWorksheet.GetFirstCellOfRow(ARow: Cardinal): PCell; var c, n: Cardinal; begin n := GetLastColIndex; c := 0; Result := FindCell(ARow, c); while (result = nil) and (c < n) do begin inc(c); result := FindCell(ARow, c); end; end; {@@ Finds the last cell with contents in a given row @param ARow Index of the row considered @return Pointer to the last cell in this row, or nil if the row is empty. } function TsWorksheet.GetLastCellOfRow(ARow: Cardinal): PCell; var c, n: Cardinal; begin n := GetLastColIndex; c := n; Result := FindCell(ARow, c); while (Result = nil) and (c > 0) do begin dec(c); Result := FindCell(ARow, c); end; end; {@@ Returns the 0-based index of the last row with a cell with contents. If no cells have contents, zero will be returned, which is also a valid value. Use GetCellCount to verify if there is at least one cell with contents in the worksheet. @see GetCellCount } function TsWorksheet.GetLastRowIndex: Cardinal; var AVLNode: TAVLTreeNode; i: Integer; begin Result := 0; AVLNode := FCells.FindHighest; if Assigned(AVLNode) then Result := PCell(AVLNode.Data).Row; // In addition, there may be row records even for empty rows. for i:=0 to FRows.Count-1 do if FRows[i] <> nil then Result := Math.Max(Result, PRow(FRows[i])^.Row); end; {@@ Deprecated, use GetLastColIndex instead @see GetLastColIndex } function TsWorksheet.GetLastRowNumber: Cardinal; begin Result := GetLastRowIndex; end; {@@ Reads the contents of a cell and returns an user readable text representing the contents of the cell. The resulting ansistring is UTF-8 encoded. @param ARow The row of the cell @param ACol The column of the cell @return The text representation of the cell } function TsWorksheet.ReadAsUTF8Text(ARow, ACol: Cardinal): ansistring; begin Result := ReadAsUTF8Text(GetCell(ARow, ACol)); end; {@@ Reads the contents of a cell and returns an user readable text representing the contents of the cell. The resulting ansistring is UTF-8 encoded. @param ACell Pointer to the cell @return The text representation of the cell } function TsWorksheet.ReadAsUTF8Text(ACell: PCell): ansistring; function FloatToStrNoNaN(const Value: Double; ANumberFormat: TsNumberFormat; ANumberFormatStr: string): ansistring; var fs: TFormatSettings; left, right: String; begin fs := FWorkbook.FormatSettings; if IsNan(Value) then Result := '' else if (ANumberFormat = nfGeneral) or (ANumberFormatStr = '') then Result := FloatToStr(Value, fs) else if ANumberFormat = nfSci then Result := SciFloat(Value, CountDecs(ANumberFormatStr, ['0']), fs) else if (ANumberFormat = nfPercentage) then Result := FormatFloat(ANumberFormatStr, Value*100, fs) else if (ANumberFormat in [nfAccounting, nfAccountingRed]) then case SplitAccountingFormatString(ANumberFormatStr, Sign(Value), left, right) of 0: Result := FormatFloat(ANumberFormatStr, Value, fs); 1: Result := FormatFloat(left, abs(Value), fs) + ' ' + Right; 2: Result := Left + ' ' + FormatFloat(right, abs(Value), fs); end else Result := FormatFloat(ANumberFormatStr, Value, fs) end; function DateTimeToStrNoNaN(const Value: Double; ANumberFormat: TsNumberFormat; ANumberFormatStr: String): ansistring; var fmtp, fmtn, fmt0: String; begin Result := ''; if not IsNaN(Value) then begin if ANumberFormatStr = '' then ANumberFormatStr := BuildDateTimeFormatString(ANumberFormat, Workbook.FormatSettings, ANumberFormatStr); // Saw strange cases in ods where date/time formats contained pos/neg/zero parts. // Split to be on the safe side. SplitFormatString(ANumberFormatStr, fmtp, fmtn, fmt0); if (Value > 0) or ((Value = 0) and (fmt0 = '')) or ((Value < 0) and (fmtn = '')) then Result := FormatDateTime(fmtp, Value, [fdoInterval]) else if (Value < 0) then Result := FormatDateTime(fmtn, Value, [fdoInterval]) else if (Value = 0) then Result := FormatDateTime(fmt0, Value, [fdoInterval]); end; end; begin Result := ''; if ACell = nil then Exit; with ACell^ do case ContentType of cctNumber: Result := FloatToStrNoNaN(NumberValue, NumberFormat, NumberFormatStr); cctUTF8String: Result := UTF8StringValue; cctDateTime: Result := DateTimeToStrNoNaN(DateTimeValue, NumberFormat, NumberFormatStr); cctBool: Result := StrUtils.IfThen(BoolValue, lpTRUE, lpFALSE); cctError: case TsErrorValue(ErrorValue) of errEmptyIntersection : Result := lpErrEmptyIntersection; errDivideByZero : Result := lpErrDivideByZero; errWrongType : Result := lpErrWrongType; errIllegalRef : Result := lpErrIllegalRef; errWrongName : Result := lpErrWrongName; errOverflow : Result := lpErrOverflow; errArgError : Result := lpErrArgError; errFormulaNotSupported: Result := lpErrFormulaNotSupported; end; else Result := ''; end; end; {@@ Returns the value of a cell as a number. If the cell contains a date/time value its serial value is returned (as FPC TDateTime). If the cell contains a text value it is attempted to convert it to a number. If the cell is empty or its contents cannot be represented as a number the value 0.0 is returned. @param ARow The row of the cell @param ACol The column of the cell @return Floating-point value representing the cell contents, or 0.0 if cell does not exist or its contents cannot be converted to a number. } function TsWorksheet.ReadAsNumber(ARow, ACol: Cardinal): Double; begin Result := ReadAsNumber(FindCell(ARow, ACol)); end; {@@ Returns the value of a cell as a number. If the cell contains a date/time value its serial value is returned (as FPC TDateTime). If the cell contains a text value it is attempted to convert it to a number. If the cell is empty or its contents cannot be represented as a number the value 0.0 is returned. @param ACell Pointer to the cell @return Floating-point value representing the cell contents, or 0.0 if cell does not exist or its contents cannot be converted to a number. } function TsWorksheet.ReadAsNumber(ACell: PCell): Double; begin Result := 0.0; if ACell = nil then exit; case ACell^.ContentType of cctDateTime : Result := ACell^.DateTimeValue; //this is in FPC TDateTime format, not Excel cctNumber : Result := ACell^.NumberValue; cctUTF8String : if not TryStrToFloat(ACell^.UTF8StringValue, Result) then Result := 0.0; cctBool : if ACell^.BoolValue then Result := 1.0 else Result := 0.0; end; end; {@@ Reads the contents of a cell and returns the date/time value of the cell. @param ARow The row of the cell @param ACol The column of the cell @param AResult Date/time value of the cell (or 0.0, if no date/time cell) @return True if the cell is a datetime value, false otherwise } function TsWorksheet.ReadAsDateTime(ARow, ACol: Cardinal; out AResult: TDateTime): Boolean; begin Result := ReadAsDateTime(FindCell(ARow, ACol), AResult); end; {@@ Reads the contents of a cell and returns the date/time value of the cell. @param ACell Pointer to the cell @param AResult Date/time value of the cell (or 0.0, if no date/time cell) @return True if the cell is a datetime value, false otherwise } function TsWorksheet.ReadAsDateTime(ACell: PCell; out AResult: TDateTime): Boolean; begin if (ACell = nil) or (ACell^.ContentType <> cctDateTime) then begin AResult := 0; Result := False; Exit; end; AResult := ACell^.DateTimeValue; Result := True; end; {@@ If a cell contains a formula (string formula or RPN formula) the formula is returned as a string in Excel syntax. @param ACell Pointer to the cell considered @return Formula string in Excel syntax. } function TsWorksheet.ReadFormulaAsString(ACell: PCell): String; begin Result := ''; if ACell = nil then exit; if Length(ACell^.RPNFormulaValue) > 0 then Result := ReadRPNFormulaAsString(ACell) else Result := ACell^.FormulaValue.FormulaStr; end; {@@ If a cell contains an RPN formula an Excel-like formula string is constructed and returned. @param ACell Pointer to the cell considered @return Formula string in Excel syntax. } function TsWorksheet.ReadRPNFormulaAsString(ACell: PCell): String; var fs: TFormatSettings; elem: TsFormulaElement; i, j: Integer; L: TStringList; s: String; ptr: Pointer; fek: TFEKind; begin Result := ''; if ACell = nil then exit; fs := Workbook.FormatSettings; L := TStringList.Create; try for i:=0 to Length(ACell^.RPNFormulaValue)-1 do begin elem := ACell^.RPNFormulaValue[i]; ptr := Pointer(elem.ElementKind); case elem.ElementKind of fekNum: L.AddObject(Format('%g', [elem.DoubleValue], fs), ptr); fekInteger: L.AddObject(IntToStr(elem.IntValue), ptr); fekString: L.AddObject('"' + elem.StringValue + '"', ptr); fekBool: L.AddObject(IfThen(boolean(elem.IntValue), 'FALSE', 'TRUE'), ptr); fekCell, fekCellRef: L.AddObject(GetCellString(elem.Row, elem.Col, elem.RelFlags), ptr); fekCellRange: L.AddObject(GetCellRangeString(elem.Row, elem.Col, elem.Row2, elem.Col2, elem.RelFlags), ptr); // Operations: else L.AddObject(FEProps[elem.ElementKind].Symbol, ptr); end; end; i := L.Count-1; while (L.Count > 0) and (i >= 0) do begin fek := TFEKind(PtrInt(L.Objects[i])); case fek of fekAdd, fekSub, fekMul, fekDiv, fekPower, fekConcat, fekEqual, fekNotEqual, fekLess, fekLessEqual, fekGreater, fekGreaterEqual: if i+2 < L.Count then begin L.Strings[i] := Format('%s%s%s', [L[i+2], L[i], L[i+1]]); L.Delete(i+2); L.Delete(i+1); L.Objects[i] := pointer(fekString); end else begin Result := '=' + lpErrArgError; exit; end; fekUPlus, fekUMinus: if i+1 < L.Count then begin L.Strings[i] := L[i]+L[i+1]; L.Delete(i+1); L.Objects[i] := Pointer(fekString); end else begin Result := '=' + lpErrArgError; exit; end; fekPercent: if i+1 < L.Count then begin L.Strings[i] := L[i+1]+L[i]; L.Delete(i+1); L.Objects[i] := Pointer(fekString); end else begin Result := '=' + lpErrArgError; exit; end; fekParen: if i+1 < L.Count then begin L.Strings[i] := Format('(%s)', [L[i+1]]); L.Delete(i+1); L.Objects[i] := pointer(fekString); end else begin Result := '=' + lpErrArgError; exit; end; else if fek >= fekAdd then begin elem := ACell^.RPNFormulaValue[i]; s := ''; for j:= i+elem.ParamsNum downto i+1 do begin if j < L.Count then begin s := s + fs.ListSeparator + ' ' + L[j]; L.Delete(j); end else begin Result := '=' + lpErrArgError; exit; end; end; Delete(s, 1, 2); L.Strings[i] := Format('%s(%s)', [L[i], s]); L.Objects[i] := pointer(fekString); end; end; dec(i); end; if L.Count > 1 then Result := '=' + lpErrArgError // too many arguments else Result := '=' + L[0]; finally L.Free; end; end; {@@ Reads the set of used formatting fields of a cell. Each cell contains a set of "used formatting fields". Formatting is applied only if the corresponding element is contained in the set. @param ARow Row index of the considered cell @param ACol Column index of the considered cell @return Set of elements used in formatting the cell } function TsWorksheet.ReadUsedFormatting(ARow, ACol: Cardinal): TsUsedFormattingFields; var ACell: PCell; begin ACell := FindCell(ARow, ACol); if ACell = nil then begin Result := []; Exit; end; Result := ACell^.UsedFormattingFields; end; {@@ Returns the background color of a cell as index into the workbook's color palette. @param ARow Row index of the cell @param ACol Column index of the cell @return Index of the cell background color into the workbook's color palette } function TsWorksheet.ReadBackgroundColor(ARow, ACol: Cardinal): TsColor; var ACell: PCell; begin ACell := FindCell(ARow, ACol); if ACell = nil then begin Result := scWhite; Exit; end; Result := ACell^.BackgroundColor; end; {@@ Clears the list of cells and releases their memory. } procedure TsWorksheet.RemoveAllCells; var Node: TAVLTreeNode; begin Node:=FCells.FindLowest; while Assigned(Node) do begin RemoveCallback(Node.Data,nil); Node.Data:=nil; Node:=FCells.FindSuccessor(Node); end; FCells.Clear; end; {@@ Writes UTF-8 encoded text to a cell. On formats that don't support unicode, the text will be converted to ISO Latin 1. @param ARow The row of the cell @param ACol The column of the cell @param AText The text to be written encoded in utf-8 } procedure TsWorksheet.WriteUTF8Text(ARow, ACol: Cardinal; AText: ansistring); var ACell: PCell; begin ACell := GetCell(ARow, ACol); WriteUTF8Text(ACell, AText); end; {@@ Writes UTF-8 encoded text to a cell. On formats that don't support unicode, the text will be converted to ISO Latin 1. @param ACell Poiner to the cell @param AText The text to be written encoded in utf-8 } procedure TsWorksheet.WriteUTF8Text(ACell: PCell; AText: ansistring); begin if ACell = nil then exit; ACell^.ContentType := cctUTF8String; ACell^.UTF8StringValue := AText; ChangedCell(ACell^.Row, ACell^.Col); end; {@@ Writes a floating-point number to a cell @param ARow Cell row index @param ACol Cell column index @param ANumber Number to be written @param AFormat Identifier for a built-in number format, e.g. nfFixed (optional) @param ADecimals Number of decimal places used for formatting (optional) @see TsNumberFormat } procedure TsWorksheet.WriteNumber(ARow, ACol: Cardinal; ANumber: double; AFormat: TsNumberFormat = nfGeneral; ADecimals: Byte = 2); begin WriteNumber(GetCell(ARow, ACol), ANumber, AFormat, ADecimals); end; {@@ Writes a floating-point number to a cell @param ACell Pointer to the cell @param ANumber Number to be written @param AFormat Identifier for a built-in number format, e.g. nfFixed (optional) @param ADecimals Number of decimal places used for formatting (optional) @see TsNumberFormat } procedure TsWorksheet.WriteNumber(ACell: PCell; ANumber: Double; AFormat: TsNumberFormat = nfGeneral; ADecimals: Byte = 2); begin if IsDateTimeFormat(AFormat) or IsCurrencyFormat(AFormat) then raise Exception.Create(lpInvalidNumberFormat); if ACell <> nil then begin ACell^.ContentType := cctNumber; ACell^.NumberValue := ANumber; if AFormat <> nfGeneral then begin Include(ACell^.UsedFormattingFields, uffNumberFormat); ACell^.NumberFormat := AFormat; ACell^.NumberFormatStr := BuildNumberFormatString(ACell^.NumberFormat, Workbook.FormatSettings, ADecimals); end else begin Exclude(ACell^.UsedFormattingFields, uffNumberFormat); ACell^.NumberFormat := nfGeneral; ACell^.NumberFormatStr := ''; end; ChangedCell(ACell^.Row, ACell^.Col); end; end; {@@ Writes a floating point number to the cell and uses a custom number format specified by the format string. Note that fpspreadsheet may not be able to detect the formatting when reading the file. @param ARow Cell row index @param ACol Cell column index @param ANumber Number to be written @param AFormat Format identifier (nfCustom) @param AFormatString String of formatting codes (such as 'dd/mmm' } procedure TsWorksheet.WriteNumber(ARow, ACol: Cardinal; ANumber: Double; AFormat: TsNumberFormat; AFormatString: String); begin WriteNumber(GetCell(ARow, ACol), ANumber, AFormat, AFormatString); end; {@@ Writes a floating point number to the cell and uses a custom number format specified by the format string. Note that fpspreadsheet may not be able to detect the formatting when reading the file. @param ACell Pointer to the cell considered @param ANumber Number to be written @param AFormat Format identifier (nfCustom) @param AFormatString String of formatting codes (such as 'dd/mmm' } procedure TsWorksheet.WriteNumber(ACell: PCell; ANumber: Double; AFormat: TsNumberFormat; AFormatString: String); var parser: TsNumFormatParser; begin if ACell <> nil then begin parser := TsNumFormatParser.Create(Workbook, AFormatString); try // Format string ok? if parser.Status <> psOK then raise Exception.Create(lpNoValidNumberFormatString); // Make sure that we do not write a date/time value here if parser.IsDateTimeFormat then raise Exception.Create(lpInvalidNumberFormat); // If format string matches a built-in format use its format identifier, // All this is considered when calling Builtin_NumFormat of the parser. finally parser.Free; end; Include(ACell^.UsedFormattingFields, uffNumberFormat); ACell^.ContentType := cctNumber; ACell^.NumberValue := ANumber; ACell^.NumberFormat := AFormat; //nfCustom; ACell^.NumberFormatStr := AFormatString; ChangedCell(ACell^.Row, ACell^.Col); end; end; {@@ Writes as empty cell @param ARow The row of the cell @param ACol The column of the cell Note: Empty cells are useful when, for example, a border line extends along a range of cells including empty cells. } procedure TsWorksheet.WriteBlank(ARow, ACol: Cardinal); var ACell: PCell; begin ACell := GetCell(ARow, ACol); ACell^.ContentType := cctEmpty; ChangedCell(ARow, ACol); end; {@@ Writes as boolean cell @param ARow The row of the cell @param ACol The column of the cell @param AValue The boolean value } procedure TsWorksheet.WriteBoolValue(ARow, ACol: Cardinal; AValue: Boolean); var ACell: PCell; begin ACell := GetCell(ARow, ACol); ACell^.ContentType := cctBool; ACell^.BoolValue := AValue; ChangedCell(ARow, ACol); end; {@@ Writes a currency value to a given cell. Its number format can be provided optionally by specifying various parameters. @param ARow Cell row index @param ACol Cell column index @param AValue Number value to be written @param AFormat Format identifier, must be nfCurrency, nfCurrencyRed, nfAccounting, or nfAccountingRed @param ADecimals Number of decimal places @param APosCurrFormat Code specifying the order of value, currency symbol and spaces (see pcfXXXX constants) @param ANegCurrFormat Code specifying the order of value, currency symbol, spaces, and how negative values are shown (see ncfXXXX constants) @param ACurrencySymbol String to be shown as currency, such as '$', or 'EUR'. In case of '?' the currency symbol defined in the workbook's FormatSettings is used. } procedure TsWorksheet.WriteCurrency(ARow, ACol: Cardinal; AValue: Double; AFormat: TsNumberFormat = nfCurrency; ADecimals: Integer = 2; ACurrencySymbol: String = '?'; APosCurrFormat: Integer = -1; ANegCurrFormat: Integer = -1); begin WriteCurrency(GetCell(ARow, ACol), AValue, AFormat, ADecimals, ACurrencySymbol, APosCurrFormat, ANegCurrFormat); end; {@@ Writes a currency value to a given cell. Its number format can be provided optionally by specifying various parameters. @param ACell Pointer to the cell considered @param AValue Number value to be written @param AFormat Format identifier, must be nfCurrency, nfCurrencyRed, nfAccounting, or nfAccountingRed @param ADecimals Number of decimal places @param APosCurrFormat Code specifying the order of value, currency symbol and spaces (see pcfXXXX constants) @param ANegCurrFormat Code specifying the order of value, currency symbol, spaces, and how negative values are shown (see ncfXXXX constants) @param ACurrencySymbol String to be shown as currency, such as '$', or 'EUR'. In case of '?' the currency symbol defined in the workbook's FormatSettings is used. } procedure TsWorksheet.WriteCurrency(ACell: PCell; AValue: Double; AFormat: TsNumberFormat = nfCurrency; ADecimals: Integer = -1; ACurrencySymbol: String = '?'; APosCurrFormat: Integer = -1; ANegCurrFormat: Integer = -1); var fmt: String; begin if ADecimals = -1 then ADecimals := Workbook.FormatSettings.CurrencyDecimals; if APosCurrFormat = -1 then APosCurrFormat := Workbook.FormatSettings.CurrencyFormat; if ANegCurrFormat = -1 then ANegCurrFormat := Workbook.FormatSettings.NegCurrFormat; if ACurrencySymbol = '?' then ACurrencySymbol := AnsiToUTF8(Workbook.FormatSettings.CurrencyString); fmt := BuildCurrencyFormatString( nfdDefault, AFormat, Workbook.FormatSettings, ADecimals, APosCurrFormat, ANegCurrFormat, ACurrencySymbol); WriteCurrency(ACell, AValue, AFormat, fmt); end; {@@ Writes a currency value to a given cell. Its number format is specified by means of a format string. @param ARow Cell row index @param ACol Cell column index @param AValue Number value to be written @param AFormat Format identifier, must be nfCurrency, nfCurrencyRed, nfAccounting, or nfAccountingRed @param AFormatString String of formatting codes, including currency symbol. Can contain sections for different formatting of positive and negative number. Example: '"EUR" #,##0.00;("EUR" #,##0.00)' } procedure TsWorksheet.WriteCurrency(ARow, ACol: Cardinal; AValue: Double; AFormat: TsNumberFormat; AFormatString: String); begin WriteCurrency(GetCell(ARow, ACol), AValue, AFormat, AFormatString); end; {@@ Writes a currency value to a given cell. Its number format is specified by means of a format string. @param ACell Pointer to the cell considered @param AValue Number value to be written @param AFormat Format identifier, must be nfCurrency, nfCurrencyRed, nfAccounting, or nfAccountingRed @param AFormatString String of formatting codes, including currency symbol. Can contain sections for different formatting of positive and negative number. Example: '"EUR" #,##0.00;("EUR" #,##0.00)' } procedure TsWorksheet.WriteCurrency(ACell: PCell; AValue: Double; AFormat: TsNumberFormat; AFormatString: String); begin if (ACell <> nil) and IsCurrencyFormat(AFormat) then begin Include(ACell^.UsedFormattingFields, uffNumberFormat); ACell^.ContentType := cctNumber; ACell^.NumberValue := AValue; ACell^.NumberFormat := AFormat; ACell^.NumberFormatStr := AFormatString; ChangedCell(ACell^.Row, ACell^.Col); end; end; {@@ Writes a date/time value to a cell @param ARow The row of the cell @param ACol The column of the cell @param AValue The date/time/datetime to be written @param AFormat The format specifier, e.g. nfShortDate (optional) If not specified format is not changed. @param AFormatStr Format string, used only for nfCustom or nfTimeInterval. Note: at least Excel xls does not recognize a separate datetime cell type: a datetime is stored as a (floating point) number, and the cell is formatted as a date (either built-in or a custom format). } procedure TsWorksheet.WriteDateTime(ARow, ACol: Cardinal; AValue: TDateTime; AFormat: TsNumberFormat = nfShortDateTime; AFormatStr: String = ''); begin WriteDateTime(GetCell(ARow, ACol), AValue, AFormat, AFormatStr); end; {@@ Writes a date/time value to a cell @param ACell Pointer to the cell considered @param AValue The date/time/datetime to be written @param AFormat The format specifier, e.g. nfShortDate (optional) If not specified format is not changed. @param AFormatStr Format string, used only for nfCustom or nfTimeInterval. Note: at least Excel xls does not recognize a separate datetime cell type: a datetime is stored as a (floating point) number, and the cell is formatted as a date (either built-in or a custom format). } procedure TsWorksheet.WriteDateTime(ACell: PCell; AValue: TDateTime; AFormat: TsNumberFormat = nfShortDateTime; AFormatStr: String = ''); var parser: TsNumFormatParser; begin if ACell <> nil then begin ACell^.ContentType := cctDateTime; ACell^.DateTimeValue := AValue; // Date/time is actually a number field in Excel. // To make sure it gets saved correctly, set a date format (instead of General). // The user can choose another date format if he wants to if AFormatStr = '' then AFormatStr := BuildDateTimeFormatString(AFormat, Workbook.FormatSettings, AFormatStr) else if AFormat = nfTimeInterval then AFormatStr := AddIntervalBrackets(AFormatStr); // Check whether the formatstring is for date/times. if AFormatStr <> '' then begin parser := TsNumFormatParser.Create(Workbook, AFormatStr); try // Format string ok? if parser.Status <> psOK then raise Exception.Create(lpNoValidNumberFormatString); // Make sure that we do not use a number format for date/times values. if not parser.IsDateTimeFormat then raise Exception.Create(lpInvalidDateTimeFormat); // Avoid possible duplication of standard formats if AFormat = nfCustom then AFormat := parser.NumFormat; finally parser.Free; end; end; Include(ACell^.UsedFormattingFields, uffNumberFormat); ACell^.NumberFormat := AFormat; ACell^.NumberFormatStr := AFormatStr; ChangedCell(ACell^.Row, ACell^.Col); end; end; {@@ Writes a date/time value to a cell @param ARow The row index of the cell @param ACol The column index of the cell @param AValue The date/time/datetime to be written @param AFormatStr Format string (the format identifier nfCustom is used to classify the format). Note: at least Excel xls does not recognize a separate datetime cell type: a datetime is stored as a (floating point) number, and the cell is formatted as a date (either built-in or a custom format). } procedure TsWorksheet.WriteDateTime(ARow, ACol: Cardinal; AValue: TDateTime; AFormatStr: String); begin WriteDateTime(GetCell(ARow, ACol), AValue, AFormatStr); end; {@@ Writes a date/time value to a cell @param ACell Pointer to the cell considered @param AValue The date/time/datetime to be written @param AFormatStr Format string (the format identifier nfCustom is used to classify the format). Note: at least Excel xls does not recognize a separate datetime cell type: a datetime is stored as a (floating point) number, and the cell is formatted as a date (either built-in or a custom format). } procedure TsWorksheet.WriteDateTime(ACell: PCell; AValue: TDateTime; AFormatStr: String); begin WriteDateTime(ACell, AValue, nfCustom, AFormatStr); end; {@@ Formats the number in a cell to show a given count of decimal places. Is ignored for non-decimal formats (such as most date/time formats). @param ARow Row indows of the cell considered @param ACol Column indows of the cell considered @param ADecimals Number of decimal places to be displayed @see TsNumberFormat } procedure TsWorksheet.WriteDecimals(ARow, ACol: Cardinal; ADecimals: Byte); begin WriteDecimals(FindCell(ARow, ACol), ADecimals); end; {@@ Formats the number in a cell to show a given count of decimal places. Is ignored for non-decimal formats (such as most date/time formats). @param ACell Pointer to the cell considered @param ADecimals Number of decimal places to be displayed @see TsNumberFormat } procedure TsWorksheet.WriteDecimals(ACell: PCell; ADecimals: Byte); var parser: TsNumFormatParser; begin if (ACell <> nil) and (ACell^.ContentType = cctNumber) and (ACell^.NumberFormat <> nfCustom) then begin parser := TsNumFormatParser.Create(Workbook, ACell^.NumberFormatStr); try parser.Decimals := ADecimals; ACell^.NumberFormatStr := parser.FormatString[nfdDefault]; finally parser.Free; end; ChangedCell(ACell^.Row, ACell^.Col); end; end; {@@ Writes an error value to a cell. @param ARow The row of the cell @param ACol The column of the cell @param AValue The error code value @see TsErrorValue } procedure TsWorksheet.WriteErrorValue(ARow, ACol: Cardinal; AValue: TsErrorValue); begin WriteErrorValue(GetCell(ARow, ACol), AValue); end; {@@ Writes an error value to a cell. @param ACol Pointer to the cell considered @param AValue The error code value @see TsErrorValue } procedure TsWorksheet.WriteErrorValue(ACell: PCell; AValue: TsErrorValue); begin if ACell <> nil then begin ACell^.ContentType := cctError; ACell^.ErrorValue := AValue; ChangedCell(ACell^.Row, ACell^.Col); end; end; {@@ Writes a formula to a given cell @param ARow The row of the cell @param ACol The column of the cell @param AFormula The formula to be written } procedure TsWorksheet.WriteFormula(ARow, ACol: Cardinal; AFormula: TsFormula); var ACell: PCell; begin ACell := GetCell(ARow, ACol); ACell^.ContentType := cctFormula; ACell^.FormulaValue := AFormula; ChangedCell(ARow, ACol); end; {@@ Adds number format to the formatting of a cell @param ARow The row of the cell @param ACol The column of the cell @param ANumberFormat Identifier of the format to be applied @param AFormatString optional string of formatting codes. Is only considered if ANumberFormat is nfCustom. @see TsNumberFormat } procedure TsWorksheet.WriteNumberFormat(ARow, ACol: Cardinal; ANumberFormat: TsNumberFormat; const AFormatString: String = ''); var ACell: PCell; begin ACell := GetCell(ARow, ACol); WriteNumberFormat(ACell, ANumberFormat, AFormatString); end; {@@ Adds a number format to the formatting of a cell @param ACell Pointer to the cell considered @param ANumberFormat Identifier of the format to be applied @param AFormatString optional string of formatting codes. Is only considered if ANumberFormat is nfCustom. @see TsNumberFormat } procedure TsWorksheet.WriteNumberFormat(ACell: PCell; ANumberFormat: TsNumberFormat; const AFormatString: String = ''); begin if ACell = nil then exit; Include(ACell^.UsedFormattingFields, uffNumberFormat); ACell^.NumberFormat := ANumberFormat; if (AFormatString = '') then ACell^.NumberFormatStr := BuildNumberFormatString(ANumberFormat, Workbook.FormatSettings) else ACell^.NumberFormatStr := AFormatString; ChangedCell(ACell^.Row, ACell^.Col); end; {@@ Writes an RPN formula to a cell. An RPN formula is an array of tokens describing the calculation to be performed. @param ARow Row indows of the cell considered @param ACol Column index of the cell @param AFormula Array of TsFormulaElements. The array can be created by using "CreateRPNFormla". @see TsNumberFormat @see TsFormulaElements @see CreateRPNFormula } procedure TsWorksheet.WriteRPNFormula(ARow, ACol: Cardinal; AFormula: TsRPNFormula); var ACell: PCell; begin ACell := GetCell(ARow, ACol); ACell^.ContentType := cctRPNFormula; ACell^.RPNFormulaValue := AFormula; ChangedCell(ARow, ACol); end; {@@ Adds font specification to the formatting of a cell. Looks in the workbook's FontList and creates an new entry if the font is not used so far. Returns the index of the font in the font list. @param ARow The row of the cell @param ACol The column of the cell @param AFontName Name of the font @param AFontSize Size of the font, in points @param AFontStyle Set with font style attributes (don't use those of unit "graphics" !) @return Index of the font in the workbook's font list. } function TsWorksheet.WriteFont(ARow, ACol: Cardinal; const AFontName: String; AFontSize: Single; AFontStyle: TsFontStyles; AFontColor: TsColor): Integer; var lCell: PCell; begin lCell := GetCell(ARow, ACol); Include(lCell^.UsedFormattingFields, uffFont); Result := FWorkbook.FindFont(AFontName, AFontSize, AFontStyle, AFontColor); if Result = -1 then result := FWorkbook.AddFont(AFontName, AFontSize, AFontStyle, AFontColor); lCell^.FontIndex := Result; ChangedFont(ARow, ACol); end; {@@ Applies a font to the formatting of a cell. The font is determined by its index in the workbook's font list: @param ARow The row of the cell @param ACol The column of the cell @param AFontIndex Index of the font in the workbook's font list } procedure TsWorksheet.WriteFont(ARow, ACol: Cardinal; AFontIndex: Integer); var lCell: PCell; begin if (AFontIndex >= 0) and (AFontIndex < Workbook.GetFontCount) and (AFontIndex <> 4) // note: Font index 4 is not defined in BIFF then begin lCell := GetCell(ARow, ACol); Include(lCell^.UsedFormattingFields, uffFont); lCell^.FontIndex := AFontIndex; ChangedFont(ARow, ACol); end else raise Exception.Create(lpInvalidFontIndex); end; {@@ Replaces the text color used in formatting of a cell. Looks in the workbook's font list if this modified font has already been used. If not a new font entry is created. Returns the index of this font in the font list. @param ARow The row of the cell @param ACol The column of the cell @param AFontColor Index into the workbook's color palette identifying the new text color. @return Index of the font in the workbook's font list. } function TsWorksheet.WriteFontColor(ARow, ACol: Cardinal; AFontColor: TsColor): Integer; var lCell: PCell; fnt: TsFont; begin lCell := GetCell(ARow, ACol); fnt := Workbook.GetFont(lCell^.FontIndex); Result := WriteFont(ARow, ACol, fnt.FontName, fnt.Size, fnt.Style, AFontColor); end; {@@ Replaces the font used in formatting of a cell considering only the font face and leaving font size, style and color unchanged. Looks in the workbook's font list if this modified font has already been used. If not a new font entry is created. Returns the index of this font in the font list. @param ARow The row of the cell @param ACol The column of the cell @param AFontName Name of the new font to be used @return Index of the font in the workbook's font list. } function TsWorksheet.WriteFontName(ARow, ACol: Cardinal; AFontName: String): Integer; var lCell: PCell; fnt: TsFont; begin lCell := GetCell(ARow, ACol); fnt := Workbook.GetFont(lCell^.FontIndex); result := WriteFont(ARow, ACol, AFontName, fnt.Size, fnt.Style, fnt.Color); end; {@@ Replaces the font size in formatting of a cell. Looks in the workbook's font list if this modified font has already been used. If not a new font entry is created. Returns the index of this font in the font list. @param ARow The row of the cell @param ACol The column of the cell @param ASize Size of the font to be used (in points). @return Index of the font in the workbook's font list. } function TsWorksheet.WriteFontSize(ARow, ACol: Cardinal; ASize: Single): Integer; var lCell: PCell; fnt: TsFont; begin lCell := GetCell(ARow, ACol); fnt := Workbook.GetFont(lCell^.FontIndex); Result := WriteFont(ARow, ACol, fnt.FontName, ASize, fnt.Style, fnt.Color); end; {@@ Replaces the font style (bold, italic, etc) in formatting of a cell. Looks in the workbook's font list if this modified font has already been used. If not a new font entry is created. Returns the index of this font in the font list. @param ARow The row of the cell @param ACol The column of the cell @param AStyle New font style to be used @return Index of the font in the workbook's font list. @see TsFontStyle } function TsWorksheet.WriteFontStyle(ARow, ACol: Cardinal; AStyle: TsFontStyles): Integer; var lCell: PCell; fnt: TsFont; begin lCell := GetCell(ARow, ACol); fnt := Workbook.GetFont(lCell^.FontIndex); Result := WriteFont(ARow, ACol, fnt.FontName, fnt.Size, AStyle, fnt.Color); end; {@@ Adds text rotation to the formatting of a cell @param ARow The row of the cell @param ACol The column of the cell @param ARotation How to rotate the text @see TsTextRotation } procedure TsWorksheet.WriteTextRotation(ARow, ACol: Cardinal; ARotation: TsTextRotation); var ACell: PCell; begin ACell := GetCell(ARow, ACol); Include(ACell^.UsedFormattingFields, uffTextRotation); ACell^.TextRotation := ARotation; ChangedFont(ARow, ACol); end; {@@ Directly modifies the used formatting fields of a cell. Only formatting corresponding to items included in this set is executed. @param ARow The row of the cell @param ACol The column of the cell @param AUsedFormatting set of the used formatting fields @see TsUsedFormattingFields @see TCell } procedure TsWorksheet.WriteUsedFormatting(ARow, ACol: Cardinal; AUsedFormatting: TsUsedFormattingFields); var ACell: PCell; begin ACell := GetCell(ARow, ACol); ACell^.UsedFormattingFields := AUsedFormatting; ChangedCell(ARow, ACol); end; {@@ Sets the color of a background color of a cell. @param ARow Row index of the cell @param ACol Column index of the cell @param AColor Index of the new background color into the workbook's color palette. } procedure TsWorksheet.WriteBackgroundColor(ARow, ACol: Cardinal; AColor: TsColor); var ACell: PCell; begin ACell := GetCell(ARow, ACol); ACell^.UsedFormattingFields := ACell^.UsedFormattingFields + [uffBackgroundColor]; ACell^.BackgroundColor := AColor; ChangedCell(ARow, ACol); end; {@@ Sets the color of a cell border line. Note: the border must be included in Borders set in order to be shown! @param ARow Row index of the cell @param ACol Column index of the cell @param ABorder Indicates to which border (left/top etc) this color is to be applied @param AColor Index of the new border color into the workbook's color palette. } procedure TsWorksheet.WriteBorderColor(ARow, ACol: Cardinal; ABorder: TsCellBorder; AColor: TsColor); var lCell: PCell; begin lCell := GetCell(ARow, ACol); lCell^.BorderStyles[ABorder].Color := AColor; ChangedCell(ARow, ACol); end; {@@ Sets the linestyle of a cell border. Note: the border must be included in the "Borders" set in order to be shown! @param ARow Row index of the cell @param ACol Column index of the cell @param ABorder Indicates to which border (left/top etc) this color is to be applied @param ALineStyle Identifier of the new line style to be applied. @see TsLineStyle } procedure TsWorksheet.WriteBorderLineStyle(ARow, ACol: Cardinal; ABorder: TsCellBorder; ALineStyle: TsLineStyle); var lCell: PCell; begin lCell := GetCell(ARow, ACol); lCell^.BorderStyles[ABorder].LineStyle := ALineStyle; ChangedCell(ARow, ACol); end; {@@ Shows the cell borders included in the set ABorders. No border lines are drawn for those not included. The borders are drawn using the "BorderStyles" assigned to the cell. @param ARow Row index of the cell @param ACol Column index of the cell @param ABorders Set with elements to identify the border(s) to will be shown @see TsCellBorder } procedure TsWorksheet.WriteBorders(ARow, ACol: Cardinal; ABorders: TsCellBorders); var lCell: PCell; begin lCell := GetCell(ARow, ACol); Include(lCell^.UsedFormattingFields, uffBorder); lCell^.Border := ABorders; ChangedCell(ARow, ACol); end; {@@ Sets the style of a cell border, i.e. line style and line color. Note: the border must be included in the "Borders" set in order to be shown! @param ARow Row index of the cell considered @param ACol Column index of the cell considered @param ABorder Identifies the border to be modified (left/top/right/bottom) @param AStyle record of parameters controlling how the border line is drawn (line style, line color) } procedure TsWorksheet.WriteBorderStyle(ARow, ACol: Cardinal; ABorder: TsCellBorder; AStyle: TsCellBorderStyle); var lCell: PCell; begin lCell := GetCell(ARow, ACol); lCell^.BorderStyles[ABorder] := AStyle; ChangedCell(ARow, ACol); end; {@@ Sets line style and line color of a cell border. Note: the border must be included in the "Borders" set in order to be shown! @param ARow Row index of the considered cell @param ACol Column index of the considered cell @param ABorder Identifier of the border to be modified @param ALineStyle Identifier for the new line style of the border @param AColor Palette index for the color of the border line @see WriteBorderStyles } procedure TsWorksheet.WriteBorderStyle(ARow, ACol: Cardinal; ABorder: TsCellBorder; ALineStyle: TsLinestyle; AColor: TsColor); var lCell: PCell; begin lCell := GetCell(ARow, ACol); lCell^.BorderStyles[ABorder].LineStyle := ALineStyle; lCell^.BorderStyles[ABorder].Color := AColor; ChangedCell(ARow, ACol); end; {@@ Sets the style of all cell border of a cell, i.e. line style and line color. Note: Only those borders included in the "Borders" set are shown! @param ARow Row index of the considered cell @param ACol Column index of the considered cell @param AStyles Array of CellBorderStyles for each cell border. @see WriteBorderStyle } procedure TsWorksheet.WriteBorderStyles(ARow, ACol: Cardinal; const AStyles: TsCellBorderStyles); var b: TsCellBorder; cell: PCell; begin cell := GetCell(ARow, ACol); for b in TsCellBorder do cell^.BorderStyles[b] := AStyles[b]; ChangedCell(ARow, ACol); end; {@@ Defines the horizontal alignment of text in a cell. @param ARow Row index of the cell considered @param ACol Column index of the cell considered @param AValue Parameter for horizontal text alignment (haDefault, vaLeft, haCenter, haRight) By default, texts are left-aligned, numbers and dates are right-aligned. } procedure TsWorksheet.WriteHorAlignment(ARow, ACol: Cardinal; AValue: TsHorAlignment); var lCell: PCell; begin lCell := GetCell(ARow, ACol); lCell^.UsedFormattingFields := lCell^.UsedFormattingFields + [uffHorAlign]; lCell^.HorAlignment := AValue; ChangedCell(ARow, ACol); end; {@@ Defines the vertical alignment of text in a cell. @param ARow Row index of the cell considered @param ACol Column index of the cell considered @param AValue Parameter for vertical text alignment (vaDefault, vaTop, vaCenter, vaBottom) By default, texts are bottom-aligned. } procedure TsWorksheet.WriteVertAlignment(ARow, ACol: Cardinal; AValue: TsVertAlignment); var lCell: PCell; begin lCell := GetCell(ARow, ACol); lCell^.UsedFormattingFields := lCell^.UsedFormattingFields + [uffVertAlign]; lCell^.VertAlignment := AValue; ChangedCell(ARow, ACol); end; {@@ Enables or disables the word-wrapping feature for a cell. @param ARow Row index of the cell considered @param ACol Column index of the cell considered @param AValue true = word-wrapping enabled, false = disabled. } procedure TsWorksheet.WriteWordwrap(ARow, ACol: Cardinal; AValue: boolean); var lCell: PCell; begin lCell := GetCell(ARow, ACol); if AValue then Include(lCell^.UsedFormattingFields, uffWordwrap) else Exclude(lCell^.UsedFormattingFields, uffWordwrap); ChangedCell(ARow, ACol); end; function TsWorksheet.GetFormatSettings: TFormatSettings; begin Result := FWorkbook.FormatSettings; end; {@@ Calculates the optimum height of a given row. Depends on the font size of the individual cells in the row. @param ARow Index of the row to be considered @return Row height in line count of the default font. } function TsWorksheet.CalcAutoRowHeight(ARow: Cardinal): Single; var cell: PCell; col: Integer; h0: Single; begin Result := 0; h0 := Workbook.GetDefaultFontSize; for col := 0 to GetLastColIndex do begin cell := FindCell(ARow, col); if cell <> nil then Result := Max(Result, Workbook.GetFont(cell^.FontIndex).Size / h0); end; end; {@@ Checks if a row record exists for the given row index and returns a pointer to the row record, or nil if not found @param ARow Index of the row looked for @return Pointer to the row record with this row index, or nil if not found } function TsWorksheet.FindRow(ARow: Cardinal): PRow; var LElement: TRow; AVLNode: TAVGLVLTreeNode; begin Result := nil; LElement.Row := ARow; AVLNode := FRows.Find(@LElement); if Assigned(AVLNode) then result := PRow(AVLNode.Data); end; {@@ Checks if a column record exists for the given column index and returns a pointer to the TCol record, or nil if not found @param ACol Index of the column looked for @return Pointer to the column record with this column index, or nil if not found } function TsWorksheet.FindCol(ACol: Cardinal): PCol; var LElement: TCol; AVLNode: TAVGLVLTreeNode; begin Result := nil; LElement.Col := ACol; AVLNode := FCols.Find(@LElement); if Assigned(AVLNode) then result := PCol(AVLNode.Data); end; {@@ Checks if a row record exists for the given row index and creates it if not found. @param ARow Index of the row looked for @return Pointer to the row record with this row index. It can safely be assumed that this row record exists. } function TsWorksheet.GetRow(ARow: Cardinal): PRow; begin Result := FindRow(ARow); if (Result = nil) then begin Result := GetMem(SizeOf(TRow)); FillChar(Result^, SizeOf(TRow), #0); Result^.Row := ARow; FRows.Add(Result); end; end; {@@ Checks if a column record exists for the given column index and creates it if not found. @param ACol Index of the column looked for @return Pointer to the TCol record with this column index. It can safely be assumed that this column record exists. } function TsWorksheet.GetCol(ACol: Cardinal): PCol; begin Result := FindCol(ACol); if (Result = nil) then begin Result := GetMem(SizeOf(TCol)); FillChar(Result^, SizeOf(TCol), #0); Result^.Col := ACol; FCols.Add(Result); end; end; {@@ Counts how many cells exist in the given column. Blank cells do contribute to the sum, as well as formatted cells. @param ACol Index of the column considered @return Count of cells with value or format in this column } function TsWorksheet.GetCellCountInCol(ACol: Cardinal): Cardinal; var cell: PCell; r: Cardinal; row: PRow; begin Result := 0; for r := 0 to GetLastRowIndex do begin cell := FindCell(r, ACol); if cell <> nil then inc(Result) else begin row := FindRow(r); if row <> nil then inc(Result); end; end; end; {@@ Counts how many cells exist in the given row. Blank cells do contribute to the sum, as well as formatted cell.s @param ARow Index of the row considered @return Count of cells with value or format in this row } function TsWorksheet.GetCellCountInRow(ARow: Cardinal): Cardinal; var cell: PCell; c: Cardinal; col: PCol; begin Result := 0; for c := 0 to GetLastColIndex do begin cell := FindCell(ARow, c); if cell <> nil then inc(Result) else begin col := FindCol(c); if col <> nil then inc(Result); end; end; end; {@@ Returns the width of the given column. If there is no column record then the default column width is returned. @param ACol Index of the column considered @return Width of the column (in count of "0" characters of the default font) } function TsWorksheet.GetColWidth(ACol: Cardinal): Single; var col: PCol; begin col := FindCol(ACol); if col <> nil then Result := col^.Width else Result := FWorkbook.DefaultColWidth; end; {@@ Returns the height of the given row. If there is no row record then the default row height is returned @param ARow Index of the row considered @return Height of the row (in line count of the default font). } function TsWorksheet.GetRowHeight(ARow: Cardinal): Single; var row: PRow; begin row := FindRow(ARow); if row <> nil then Result := row^.Height else //Result := CalcAutoRowHeight(ARow); Result := FWorkbook.DefaultRowHeight; end; {@@ Removes all row records from the worksheet and frees the occupied memory. Note: Cells are retained. } procedure TsWorksheet.RemoveAllRows; var Node: Pointer; i: Integer; begin for i := FRows.Count-1 downto 0 do begin Node := FRows.Items[i]; FreeMem(Node, SizeOf(TRow)); end; FRows.Clear; end; {@@ Removes all column records from the worksheet and frees the occupied memory. Note: Cells are retained. } procedure TsWorksheet.RemoveAllCols; var Node: Pointer; i: Integer; begin for i := FCols.Count-1 downto 0 do begin Node := FCols.Items[i]; FreeMem(Node, SizeOf(TCol)); end; FCols.Clear; end; {@@ Writes a row record for the row at a given index to the spreadsheet. Currently the row record contains only the row height (and the row index, of course). Creates a new row record if it does not yet exist. @param ARow Index of the row record which will be created or modified @param AData Data to be written. } procedure TsWorksheet.WriteRowInfo(ARow: Cardinal; AData: TRow); var AElement: PRow; begin AElement := GetRow(ARow); AElement^.Height := AData.Height; end; {@@ Sets the row height for a given row. Creates a new row record if it does not yet exist. @param ARow Index of the row to be considered @param AHeight Row height to be assigned to the row. The row height is expressed as the line count of the default font size. } procedure TsWorksheet.WriteRowHeight(ARow: Cardinal; AHeight: Single); var AElement: PRow; begin AElement := GetRow(ARow); AElement^.Height := AHeight; end; {@@ Writes a column record for the column at a given index to the spreadsheet. Currently the column record contains only the column width (and the column index, of course). Creates a new column record if it does not yet exist. @param ACol Index of the column record which will be created or modified @param AData Data to be written (essentially column width). } procedure TsWorksheet.WriteColInfo(ACol: Cardinal; AData: TCol); var AElement: PCol; begin AElement := GetCol(ACol); AElement^.Width := AData.Width; end; {@@ Sets the column width for a given column. Creates a new column record if it does not yet exist. @param ACol Index of the column to be considered @param AWidth Width to be assigned to the column. The column width is expressed as the count of "0" characters of the default font. } procedure TsWorksheet.WriteColWidth(ACol: Cardinal; AWidth: Single); var AElement: PCol; begin AElement := GetCol(ACol); AElement^.Width := AWidth; end; { TsWorkbook } {@@ Helper method for clearing the spreadsheet list. } procedure TsWorkbook.RemoveWorksheetsCallback(data, arg: pointer); begin Unused(arg); TsWorksheet(data).Free; end; {@@ Constructor of the workbook class. Among others, it initializes the built-in fonts, defines the default font, and sets up the FormatSettings for localization of some number formats. } constructor TsWorkbook.Create; begin inherited Create; FWorksheets := TFPList.Create; FFormat := sfExcel8; FDefaultColWidth := 12; FDefaultRowHeight := 1; FormatSettings := DefaultFormatSettings; FormatSettings.ShortDateFormat := MakeShortDateFormat(FormatSettings.ShortDateFormat); FormatSettings.LongDateFormat := MakeLongDateFormat(FormatSettings.ShortDateFormat); FFontList := TFPList.Create; SetDefaultFont('Arial', 10.0); InitFonts; end; {@@ Destructor of the workbook class } destructor TsWorkbook.Destroy; begin RemoveAllWorksheets; RemoveAllFonts; FWorksheets.Free; FFontList.Free; inherited Destroy; end; {@@ Helper method for determining the spreadsheet type from the file type extension @param AFileName Name of the file to be considered @param SheetType File format found from analysis of the extension (output) @return True if the file matches any of the known formats, false otherwise } class function TsWorkbook.GetFormatFromFileName(const AFileName: TFileName; out SheetType: TsSpreadsheetFormat): Boolean; var suffix: String; begin Result := True; suffix := Lowercase(ExtractFileExt(AFileName)); if suffix = STR_EXCEL_EXTENSION then SheetType := sfExcel8 else if suffix = STR_OOXML_EXCEL_EXTENSION then SheetType := sfOOXML else if suffix = STR_OPENDOCUMENT_CALC_EXTENSION then SheetType := sfOpenDocument else if suffix = STR_COMMA_SEPARATED_EXTENSION then SheetType := sfCSV else if suffix = STR_WIKITABLE_PIPES then SheetType := sfWikiTable_Pipes else if suffix = STR_WIKITABLE_WIKIMEDIA then SheetType := sfWikiTable_WikiMedia else Result := False; end; {@@ Convenience method which creates the correct reader object for a given spreadsheet format. } function TsWorkbook.CreateSpreadReader(AFormat: TsSpreadsheetFormat): TsCustomSpreadReader; var i: Integer; begin Result := nil; for i := 0 to Length(GsSpreadFormats) - 1 do if GsSpreadFormats[i].Format = AFormat then begin Result := GsSpreadFormats[i].ReaderClass.Create(self); Break; end; if Result = nil then raise Exception.Create(lpUnsupportedReadFormat); end; {@@ Convenience method which creates the correct writer object for a given spreadsheet format. } function TsWorkbook.CreateSpreadWriter(AFormat: TsSpreadsheetFormat): TsCustomSpreadWriter; var i: Integer; begin Result := nil; for i := 0 to Length(GsSpreadFormats) - 1 do if GsSpreadFormats[i].Format = AFormat then begin Result := GsSpreadFormats[i].WriterClass.Create(self); Break; end; if Result = nil then raise Exception.Create(lpUnsupportedWriteFormat); end; {@@ Reads the document from a file. } procedure TsWorkbook.ReadFromFile(AFileName: string; AFormat: TsSpreadsheetFormat); var AReader: TsCustomSpreadReader; begin AReader := CreateSpreadReader(AFormat); try AReader.ReadFromFile(AFileName, Self); FFormat := AFormat; finally AReader.Free; end; end; {@@ Reads the document from a file. This method will try to guess the format from the extension. In the case of the ambiguous xls extension, it will simply assume that it is BIFF8. Note that it could be BIFF2, 3, 4 or 5 too. } procedure TsWorkbook.ReadFromFile(AFileName: string); overload; var SheetType: TsSpreadsheetFormat; valid: Boolean; lException: Exception = nil; begin valid := GetFormatFromFileName(AFileName, SheetType); if valid then begin if SheetType = sfExcel8 then begin while True do begin try ReadFromFile(AFileName, SheetType); valid := True; except on E: Exception do begin if SheetType = sfExcel8 then lException := E; valid := False end; end; if valid or (SheetType = sfExcel2) then Break; SheetType := Pred(SheetType); end; // A failed attempt to read a file should bring an exception, so re-raise // the exception if necessary. We re-raise the exception brought by Excel 8, // since this is the most common format if (not valid) and (lException <> nil) then raise lException; end else ReadFromFile(AFileName, SheetType); end else raise Exception.CreateFmt(lpNoValidSpreadsheetFile, [AFileName]); end; procedure TsWorkbook.ReadFromFileIgnoringExtension(AFileName: string); var SheetType: TsSpreadsheetFormat; lException: Exception; begin SheetType := sfExcel8; while (SheetType in [sfExcel2..sfExcel8]) and (lException <> nil) do begin try Dec(SheetType); ReadFromFile(AFileName, SheetType); lException := nil; except on E: Exception do { do nothing } ; end; if lException = nil then Break; end; end; {@@ Reads the document from a seekable stream. } procedure TsWorkbook.ReadFromStream(AStream: TStream; AFormat: TsSpreadsheetFormat); var AReader: TsCustomSpreadReader; begin AReader := CreateSpreadReader(AFormat); try AReader.ReadFromStream(AStream, Self); finally AReader.Free; end; end; {@@ Writes the document to a file. If the file doesn't exist, it will be created. } procedure TsWorkbook.WriteToFile(const AFileName: string; const AFormat: TsSpreadsheetFormat; const AOverwriteExisting: Boolean = False); var AWriter: TsCustomSpreadWriter; begin AWriter := CreateSpreadWriter(AFormat); try AWriter.WriteToFile(AFileName, AOverwriteExisting); finally AWriter.Free; end; end; {@@ Writes the document to file based on the extension. If this was an earlier sfExcel type file, it will be upgraded to sfExcel8, } procedure TsWorkbook.WriteToFile(const AFileName: String; const AOverwriteExisting: Boolean); var SheetType: TsSpreadsheetFormat; valid: Boolean; begin valid := GetFormatFromFileName(AFileName, SheetType); if valid then WriteToFile(AFileName, SheetType, AOverwriteExisting) else raise Exception.Create(Format( '[TsWorkbook.WriteToFile] Attempted to save a spreadsheet by extension, but the extension %s is invalid.', [ExtractFileExt(AFileName)])); end; {@@ Writes the document to a stream } procedure TsWorkbook.WriteToStream(AStream: TStream; AFormat: TsSpreadsheetFormat); var AWriter: TsCustomSpreadWriter; begin AWriter := CreateSpreadWriter(AFormat); try AWriter.WriteToStream(AStream); finally AWriter.Free; end; end; {@@ Adds a new worksheet to the workbook It is added to the end of the list of worksheets @param AName The name of the new worksheet @return The instace of the newly created worksheet @see TsWorkbook } function TsWorkbook.AddWorksheet(AName: string): TsWorksheet; begin Result := TsWorksheet.Create; Result.Name := AName; Result.FWorkbook := Self; FWorksheets.Add(Pointer(Result)); end; {@@ Quick helper routine which returns the first worksheet @return A TsWorksheet instance if at least one is present. nil otherwise. @see TsWorkbook.GetWorksheetByIndex @see TsWorkbook.GetWorksheetByName @see TsWorksheet } function TsWorkbook.GetFirstWorksheet: TsWorksheet; begin Result := TsWorksheet(FWorksheets.First); end; {@@ Gets the worksheet with a given index The index is zero-based, so the first worksheet added has index 0, the second 1, etc. @param AIndex The index of the worksheet (0-based) @return A TsWorksheet instance if one is present at that index. nil otherwise. @see TsWorkbook.GetFirstWorksheet @see TsWorkbook.GetWorksheetByName @see TsWorksheet } function TsWorkbook.GetWorksheetByIndex(AIndex: Cardinal): TsWorksheet; begin if (integer(AIndex) < FWorksheets.Count) and (integer(AIndex)>=0) then Result := TsWorksheet(FWorksheets.Items[AIndex]) else Result := nil; end; {@@ Gets the worksheet with a given worksheet name @param AName The name of the worksheet @return A TsWorksheet instance if one is found with that name, nil otherwise. @see TsWorkbook.GetFirstWorksheet @see TsWorkbook.GetWorksheetByIndex @see TsWorksheet } function TsWorkbook.GetWorksheetByName(AName: String): TsWorksheet; var i:integer; begin Result := nil; for i:=0 to FWorksheets.Count-1 do begin if TsWorkSheet(FWorkSheets.Items[i]).Name=AName then begin Result := TsWorksheet(FWorksheets.Items[i]); exit; end; end; end; {@@ The number of worksheets on the workbook @see TsWorksheet } function TsWorkbook.GetWorksheetCount: Cardinal; begin Result := FWorksheets.Count; end; {@@ Clears the list of Worksheets and releases their memory. } procedure TsWorkbook.RemoveAllWorksheets; begin FWorksheets.ForEachCall(RemoveWorksheetsCallback, nil); end; { Font handling } {@@ Adds a font to the font list. Returns the index in the font list. @param AFontName Name of the font (like 'Arial') @param ASize Size of the font in points @param AStyle Style of the font, a combination of TsFontStyle elements @param AColor Color of the font, given by its index into the workbook's palette. @return Index of the font in the workbook's font list } function TsWorkbook.AddFont(const AFontName: String; ASize: Single; AStyle: TsFontStyles; AColor: TsColor): Integer; var fnt: TsFont; begin fnt := TsFont.Create; fnt.FontName := AFontName; fnt.Size := ASize; fnt.Style := AStyle; fnt.Color := AColor; Result := AddFont(fnt); end; {@@ Adds a font to the font list. Returns the index in the font list. @param AFont TsFont record containing all font parameters @return Index of the font in the workbook's font list } function TsWorkbook.AddFont(const AFont: TsFont): Integer; begin // Font index 4 does not exist in BIFF. Avoid that a real font gets this index. if FFontList.Count = 4 then FFontList.Add(nil); result := FFontList.Add(AFont); end; {@@ Copies a font list to the workbook's font list @param ASource Font list to be copied } procedure TsWorkbook.CopyFontList(ASource: TFPList); var fnt: TsFont; i: Integer; begin RemoveAllFonts; for i:=0 to ASource.Count-1 do begin fnt := TsFont(ASource.Items[i]); AddFont(fnt.FontName, fnt.Size, fnt.Style, fnt.Color); end; end; {@@ Checks whether the font with the given specification is already contained in the font list. Returns the index, or -1 if not found. @param AFontName Name of the font (like 'Arial') @param ASize Size of the font in points @param AStyle Style of the font, a combination of TsFontStyle elements @param AColor Color of the font, given by its index into the workbook's palette. @return Index of the font in the font list, or -1 if not found. } function TsWorkbook.FindFont(const AFontName: String; ASize: Single; AStyle: TsFontStyles; AColor: TsColor): Integer; var fnt: TsFont; begin for Result := 0 to FFontList.Count-1 do begin fnt := TsFont(FFontList.Items[Result]); if (fnt <> nil) and SameText(AFontName, fnt.FontName) and (abs(ASize - fnt.Size) < 0.001) and // careful when comparing floating point numbers (AStyle = fnt.Style) and (AColor = fnt.Color) then exit; end; Result := -1; end; {@@ Initializes the font list. In case of BIFF format, adds 5 fonts: 0: default font 1: like default font, but bold 2: like default font, but italic 3: like default font, but underlined 4: empty (due to a restriction of Excel) 5: like default font, but bold and italic } procedure TsWorkbook.InitFonts; var fntName: String; fntSize: Single; begin // Memorize old default font with TsFont(FFontList.Items[0]) do begin fntName := FontName; fntSize := Size; end; // Remove current font list RemoveAllFonts; // Build new font list SetDefaultFont(fntName, fntSize); // Default font (FONT0) AddFont(fntName, fntSize, [fssBold], scBlack); // FONT1 for uffBold AddFont(fntName, fntSize, [fssItalic], scBlack); // FONT2 (Italic) AddFont(fntName, fntSize, [fssUnderline], scBlack); // FONT3 (fUnderline) // FONT4 which does not exist in BIFF is added automatically with nil as place-holder AddFont(fntName, fntSize, [fssBold, fssItalic], scBlack); // FONT5 (bold & italic) FBuiltinFontCount := FFontList.Count; end; {@@ Clears the list of fonts and releases their memory. } procedure TsWorkbook.RemoveAllFonts; var i: Integer; fnt: TsFont; begin for i:=FFontList.Count-1 downto 0 do begin fnt := TsFont(FFontList.Items[i]); fnt.Free; FFontList.Delete(i); end; end; {@@ Defines the default font. This is the font with index 0 in the FontList. The next built-in fonts will have the same font name and size } procedure TsWorkbook.SetDefaultFont(const AFontName: String; ASize: Single); var i: Integer; begin if FFontList.Count = 0 then AddFont(AFontName, ASize, [], scBlack) else for i:=0 to FBuiltinFontCount-1 do begin if (i <> 4) and (i < FFontList.Count) then with TsFont(FFontList[i]) do begin FontName := AFontName; Size := ASize; end; end; end; {@@ Returns the default font. This is the first font (index 0) in the font list } function TsWorkbook.GetDefaultFont: TsFont; begin Result := GetFont(0); end; {@@ Returns the point size of the default font } function TsWorkbook.GetDefaultFontSize: Single; begin Result := GetFont(0).Size; end; {@@ Returns the font with the given index. @param AIndex Index of the font to be considered @return Record containing all parameters of the font (or nil if not found). } function TsWorkbook.GetFont(AIndex: Integer): TsFont; begin if (AIndex >= 0) and (AIndex < FFontList.Count) then Result := FFontList.Items[AIndex] else Result := nil; end; {@@ Returns the count of registered fonts } function TsWorkbook.GetFontCount: Integer; begin Result := FFontList.Count; end; {@@ Adds a color to the palette and returns its palette index, but only if the color does not already exist - in this case, it returns the index of the existing color entry. The color must in little-endian notation (like TColor of the graphics units) @param AColorValue Number containing the rgb code of the color to be added @return Index of the new (or already existing) color item } function TsWorkbook.AddColorToPalette(AColorValue: TsColorValue): TsColor; begin // Look look for the color. Is it already in the existing palette? if Length(FPalette) > 0 then for Result := 0 to Length(FPalette)-1 do if FPalette[Result] = AColorValue then exit; // No --> Add it to the palette. Result := Length(FPalette); SetLength(FPalette, Result+1); FPalette[Result] := AColorValue; end; {@@ Converts a fpspreadsheet color into into a string RRGGBB. Note that colors are written to xls files as ABGR (where A is 0). if the color is scRGBColor the color value is taken from the argument ARGBColor, otherwise from the palette entry for the color index. } function TsWorkbook.FPSColorToHexString(AColor: TsColor; ARGBColor: TFPColor): string; type TRgba = packed record Red, Green, Blue, A: Byte end; var colorvalue: TsColorValue; r,g,b: Byte; begin if AColor = scRGBColor then begin r := ARGBColor.Red div $100; g := ARGBColor.Green div $100; b := ARGBColor.Blue div $100; end else begin colorvalue := GetPaletteColor(AColor); r := TRgba(colorvalue).Red; g := TRgba(colorvalue).Green; b := TRgba(colorvalue).Blue; end; Result := Format('%.2x%.2x%.2x', [r, g, b]); end; {@@ Returns the name of the color pointed to by the given color index. If the name is not known the hex string is returned as RRGGBB. @param AColorIndex Palette index of the color considered @return String identifying the color (a color name or, if unknown, a string showing the rgb components } function TsWorkbook.GetColorName(AColorIndex: TsColor): string; var i: Integer; c: TsColorValue; begin // Get color rgb value c := GetPaletteColor(AColorIndex); // Find color value in default palette for i:=0 to High(DEFAULT_PALETTE) do if DEFAULT_PALETTE[i] = c then begin // if found: get the color name from the default color names array Result := DEFAULT_COLORNAMES[i]; exit; end; // if not found: construct a string from rgb byte values. Result := FPSColorToHexString(AColorIndex, colBlack); end; {@@ Reads the rgb color for the given index from the current palette. Can be type-cast to TColor for usage in GUI applications. @param AColorIndex Index of the color considered @return A number containing the rgb components in little-endian notation. } function TsWorkbook.GetPaletteColor(AColorIndex: TsColor): TsColorValue; begin if (AColorIndex >= 0) and (AColorIndex < GetPaletteSize) then begin if ((FPalette = nil) or (Length(FPalette) = 0)) then Result := DEFAULT_PALETTE[AColorIndex] else Result := FPalette[AColorIndex]; end else Result := $000000; // "black" as default end; {@@ Converts the palette color of the given index to a string that can be used in HTML code. For ODS. @param AColorIndex Index of the color considered @return A HTML-compatible string identifying the color. "Red", for example, is returned as '#FF0000'; } function TsWorkbook.GetPaletteColorAsHTMLStr(AColorIndex: TsColor): String; begin Result := ColorToHTMLColorStr(GetPaletteColor(AColorIndex)); end; {@@ Replaces a color value of the current palette by a new value. The color must be given as ABGR (little-endian), with A=0). @param AColorIndex Palette index of the color to be replaced @param AColorValue Number containing the rgb components of the new color } procedure TsWorkbook.SetPaletteColor(AColorIndex: TsColor; AColorValue: TsColorValue); begin if (AColorIndex >= 0) and (AColorIndex < GetPaletteSize) then begin if ((FPalette = nil) or (Length(FPalette) = 0)) then DEFAULT_PALETTE[AColorIndex] := AColorValue else FPalette[AColorIndex] := AColorValue; end; end; {@@ Returns the count of palette colors } function TsWorkbook.GetPaletteSize: Integer; begin if (FPalette = nil) or (Length(FPalette) = 0) then Result := High(DEFAULT_PALETTE) + 1 else Result := Length(FPalette); end; {@@ Instructs the workbook to take colors from the default palette. Is called from ODS reader because ODS does not have a palette. Without a palette the color constants (scRed etc.) would not be correct any more. } procedure TsWorkbook.UseDefaultPalette; begin UsePalette(@DEFAULT_PALETTE, Length(DEFAULT_PALETTE), false); end; {@@ Instructs the Workbook to take colors from the palette pointed to by the parameter APalette This palette is only used for writing. When reading the palette found in the file is used. @param APalette Pointer to the array of TsColorValue numbers which will become the new palette @param APaletteCount Count of numbers in the source palette @param ABigEnding If true, indicates that the source palette is in big-endian notation. The methods inverts the rgb components to little-endian which is used by fpspreadsheet internally. } procedure TsWorkbook.UsePalette(APalette: PsPalette; APaletteCount: Word; ABigEndian: Boolean); var i: Integer; begin {$IFOPT R+} {$DEFINE RNGCHECK} {$ENDIF} SetLength(FPalette, APaletteCount); if ABigEndian then for i:=0 to APaletteCount-1 do {$IFDEF RNGCHECK} {$R-} {$ENDIF} FPalette[i] := LongRGBToExcelPhysical(APalette^[i]) {$IFDEF RNGCHECK} {$R+} {$ENDIF} else for i:=0 to APaletteCount-1 do {$IFDEF RNGCHECK} {$R-} {$ENDIF} FPalette[i] := APalette^[i]; {$IFDEF RNGCHECK} {$R+} {$ENDIF} end; { TsCustomNumFormatList } {@@ Constructor of the number format list. @param AWorkbook The workbook is needed to get access to its "FormatSettings" for localization of some formatting strings. } constructor TsCustomNumFormatList.Create(AWorkbook: TsWorkbook); begin inherited Create; FWorkbook := AWorkbook; AddBuiltinFormats; end; {@@ Destructor of the number format list: clears the list and destroys the format items } destructor TsCustomNumFormatList.Destroy; begin Clear; inherited Destroy; end; {@@ Adds a number format described by the Excel format index, the ODF format name, the format string, and the built-in format identifier to the list and returns the index of the new item. @param AFormatIndex Format index to be used by Excel @param AFormatName Format name to be used by OpenDocument @param AFormatString String of formatting codes @param ANumFormat Identifier for built-in number format @return List index of the new item } function TsCustomNumFormatList.AddFormat(AFormatIndex: Integer; AFormatName, AFormatString: String; ANumFormat: TsNumberFormat): Integer; var item: TsNumFormatData; begin item := TsNumFormatData.Create; item.Index := AFormatIndex; item.Name := AFormatName; item.NumFormat := ANumFormat; item.FormatString := AFormatString; Result := inherited Add(item); end; {@@ Adds a number format described by the Excel format index, the format string, and the built-in format identifier to the list and returns the index of the new item in the format list. To be used when writing an Excel file. @param AFormatIndex Format index to be used by Excel @param AFormatString String of formatting codes @param ANumFormat Identifier for built-in number format @return Index of the new item in the format list } function TsCustomNumFormatList.AddFormat(AFormatIndex: Integer; AFormatString: String; ANumFormat: TsNumberFormat): integer; begin Result := AddFormat(AFormatIndex, '', AFormatString, ANumFormat); end; {@@ Adds a number format described by the ODF format name, the format string, and the built-in format identifier to the list and returns the index of the new item in the format list. To be used when writing an ODS file. @param AFormatName Format name to be used by OpenDocument @param AFormatString String of formatting codes @param ANumFormat Identifier for built-in number format @return Index of the new item in the format list } function TsCustomNumFormatList.AddFormat(AFormatName, AFormatString: String; ANumFormat: TsNumberFormat): Integer; begin if (AFormatString = '') and (ANumFormat <> nfGeneral) then begin Result := 0; exit; end; Result := AddFormat(FNextFormatIndex, AFormatName, AFormatString, ANumFormat); inc(FNextFormatIndex); end; {@@ Adds a number format described by the format string, and the built-in format identifier to the format list and returns the index of the new item in the list. The Excel format index and ODS format name are auto-generated. @param AFormatString String of formatting codes @param ANumFormat Identifier for built-in number format @return Index of the new item in the list } function TsCustomNumFormatList.AddFormat(AFormatString: String; ANumFormat: TsNumberFormat): Integer; begin Result := AddFormat('', AFormatString, ANumFormat); end; {@@ Adds the number format used by a given cell to the list. @param AFormatCell Pointer to a cell providing the format to be stored in the list } function TsCustomNumFormatList.AddFormat(AFormatCell: PCell): Integer; begin if AFormatCell = nil then raise Exception.Create('TsCustomNumFormat.Add: No nil pointers please'); if Count = 0 then raise Exception.Create('TsCustomNumFormatList: Error in program logics: You must provide built-in formats first.'); Result := AddFormat(FNextFormatIndex, AFormatCell^.NumberFormatStr, AFormatCell^.NumberFormat ); inc(FNextFormatIndex); end; {@@ Adds the builtin format items to the list. The formats must be specified in a way that is compatible with fpc syntax. Conversion of the formatstrings to the syntax used in the destination file can be done by calling "ConvertAfterReadung" bzw. "ConvertBeforeWriting". "AddBuiltInFormats" must be called before user items are added. Must specify FFirstFormatIndexInFile (BIFF5-8, e.g. doesn't save formats <164) and must initialize the index of the first user format (FNextFormatIndex) which is automatically incremented when adding user formats. In TsCustomNumFormatList nothing is added. } procedure TsCustomNumFormatList.AddBuiltinFormats; begin // must be overridden - see xlscommon as an example. end; {@@ Called from the reader when a format item has been read from an Excel file. Determines the number format type, format string etc and converts the format string to fpc syntax which is used directly for getting the cell text. @param AFormatIndex Excel index of the number format read from the file @param AFormatString String of formatting codes as read fromt the file. } procedure TsCustomNumFormatList.AnalyzeAndAdd(AFormatIndex: Integer; AFormatString: String); var nf: TsNumberFormat = nfGeneral; begin if FindByIndex(AFormatIndex) > -1 then exit; // Analyze & convert the format string, extract infos for internal formatting ConvertAfterReading(AFormatIndex, AFormatString, nf); // Add the new item AddFormat(AFormatIndex, AFormatString, nf); end; {@@ Clears the number format list and frees memory occupied by the format items. } procedure TsCustomNumFormatList.Clear; var i: Integer; begin for i:=0 to Count-1 do RemoveFormat(i); inherited Clear; end; {@@ Takes the format string as it is read from the file and extracts the built-in number format identifier out of it for use by fpc. The method also converts the format string to a form that can be used by fpc's FormatDateTime and FormatFloat. The method should be overridden in a class that knows knows more about the details of the spreadsheet file format. @param AFormatIndex Excel index of the number format read @param AFormatString string of formatting codes extracted from the file data @param ANumFormat identifier for built-in fpspreadsheet format extracted from the file data } procedure TsCustomNumFormatList.ConvertAfterReading(AFormatIndex: Integer; var AFormatString: String; var ANumFormat: TsNumberFormat); var parser: TsNumFormatParser; fmt: String; lFormatData: TsNumFormatData; i: Integer; begin i := FindByIndex(AFormatIndex); if i > 0 then begin lFormatData := Items[i]; fmt := lFormatData.FormatString; end else fmt := AFormatString; // Analyzes the format string and tries to convert it to fpSpreadsheet format. parser := TsNumFormatParser.Create(Workbook, fmt); try if parser.Status = psOK then begin ANumFormat := parser.NumFormat; AFormatString := parser.FormatString[nfdDefault]; end else begin // Show an error here? end; finally parser.Free; end; end; {@@ Is called before collecting all number formats of the spreadsheet and before writing them to file. Its purpose is to convert the format string as used by fpc to a format compatible with the spreadsheet file format. Nothing is changed in the TsCustomNumFormatList, the method needs to be overridden by a descendant class which known more about the details of the destination file format. Needs to be overridden by a class knowing more about the destination file format. @param AFormatString String of formatting codes. On input in fpc syntax. Is overwritten on output by format string compatible with the destination file. @param ANumFormat Identifier for built-in fpspreadsheet number format } procedure TsCustomNumFormatList.ConvertBeforeWriting(var AFormatString: String; var ANumFormat: TsNumberFormat); begin Unused(AFormatString, ANumFormat); // nothing to do here. But see, e.g., xlscommon.TsBIFFNumFormatList end; {@@ Deletes a format item from the list, and makes sure that its memory is released. @param AIndex List index of the item to be deleted. } procedure TsCustomNumFormatList.Delete(AIndex: Integer); begin RemoveFormat(AIndex); Delete(AIndex); end; {@@ Seeks a format item with the given properties and returns its list index, or -1 if not found. @param ANumFormat Built-in format identifier @param AFormatString String of formatting codes @return Index of the format item in the format list, or -1 if not found. } function TsCustomNumFormatList.Find(ANumFormat: TsNumberFormat; AFormatString: String): Integer; var item: TsNumFormatData; begin for Result := Count-1 downto 0 do begin item := Items[Result]; if (item <> nil) and (item.NumFormat = ANumFormat) and (item.FormatString = AFormatString) then exit; end; Result := -1; end; {@@ Finds the item with the given format string and returns its index in the format list, or -1 if not found. @param AFormatString string of formatting codes to be searched in the list. @return Index of the format item in the format list, or -1 if not found. } function TsCustomNumFormatList.Find(AFormatString: String): integer; var item: TsNumFormatData; begin { We search backwards to find user-defined items first. They usually are more appropriate than built-in items. } for Result := Count-1 downto 0 do begin item := Items[Result]; if item.FormatString = AFormatString then exit; end; Result := -1; end; {@@ Finds the item with the given Excel format index and returns its index in the format list, or -1 if not found. Is used by BIFF file formats. @param AFormatIndex Excel format index to the searched @return Index of the format item in the format list, or -1 if not found. } function TsCustomNumFormatList.FindByIndex(AFormatIndex: Integer): integer; var item: TsNumFormatData; begin for Result := 0 to Count-1 do begin item := Items[Result]; if item.Index = AFormatIndex then exit; end; Result := -1; end; {@@ Finds the item with the given ODS format name and returns its index in the format list (or -1, if not found) To be used by OpenDocument file format. @param AFormatName Format name as used by OpenDocument to identify a number format @return Index of the format item in the list, or -1 if not found } function TsCustomNumFormatList.FindByName(AFormatName: String): integer; var item: TsNumFormatData; begin for Result := 0 to Count-1 do begin item := Items[Result]; if item.Name = AFormatName then exit; end; Result := -1; end; {@@ Determines whether the format attributed to the given cell is already contained in the list and returns its list index, or -1 if not found @param AFormatCell Pointer to a spreadsheet cell having the number format that is looked for. @return Index of the format item in the list, or -1 if not found. } function TsCustomNumFormatList.FindFormatOf(AFormatCell: PCell): integer; begin if AFormatCell = nil then Result := -1 else Result := Find(AFormatCell^.NumberFormat, AFormatCell^.NumberFormatStr); end; {@@ Determines the format string to be written into the spreadsheet file. Calls ConvertBeforeWriting in order to convert the fpc format strings to the dialect used in the file. @param AIndex The number format at this item is considered. @return String of formatting codes that will be written to the file. } function TsCustomNumFormatList.FormatStringForWriting(AIndex: Integer): String; var item: TsNumFormatdata; nf: TsNumberFormat; begin item := Items[AIndex]; if item <> nil then begin Result := item.FormatString; nf := item.NumFormat; ConvertBeforeWriting(Result, nf); end else Result := ''; end; function TsCustomNumFormatList.GetItem(AIndex: Integer): TsNumFormatData; begin Result := TsNumFormatData(inherited Items[AIndex]); end; {@@ Deletes the memory occupied by the formatting data, but keeps an empty item in the list to retain the indexes of following items. @param AIndex The number format item at this index will be removed. } procedure TsCustomNumFormatList.RemoveFormat(AIndex: Integer); var item: TsNumFormatData; begin item := GetItem(AIndex); if item <> nil then begin item.Free; SetItem(AIndex, nil); end; end; procedure TsCustomNumFormatList.SetItem(AIndex: Integer; AValue: TsNumFormatData); begin inherited Items[AIndex] := AValue; end; function CompareNumFormatData(Item1, Item2: Pointer): Integer; begin Result := CompareValue(TsNumFormatData(Item1).Index, TsNumFormatData(Item2).Index); end; {@@ Sorts the format data items in ascending order of the Excel format indexes. } procedure TsCustomNumFormatList.Sort; begin inherited Sort(@CompareNumFormatData); end; { TsCustomSpreadReader } constructor TsCustomSpreadReader.Create(AWorkbook: TsWorkbook); begin inherited Create; FWorkbook := AWorkbook; CreateNumFormatList; // FNumFormatList.FWorkbook := AWorkbook; end; destructor TsCustomSpreadReader.Destroy; begin FNumFormatList.Free; inherited Destroy; end; procedure TsCustomSpreadReader.CreateNumFormatList; begin { The format list needs to be created by descendants who know about the special requirements of the file format. } end; {@@ Default file reading method. Opens the file and calls ReadFromStream @param AFileName The input file name. @param AData The Workbook to be filled with information from the file. @see TsWorkbook } procedure TsCustomSpreadReader.ReadFromFile(AFileName: string; AData: TsWorkbook); var InputFile: TFileStream; begin InputFile := TFileStream.Create(AFileName, fmOpenRead); try ReadFromStream(InputFile, AData); finally InputFile.Free; end; end; {@@ This routine should be overriden in descendent classes. } procedure TsCustomSpreadReader.ReadFromStream(AStream: TStream; AData: TsWorkbook); var AStringStream: TStringStream; AStrings: TStringList; begin AStringStream := TStringStream.Create(''); AStrings := TStringList.Create; try AStringStream.CopyFrom(AStream, AStream.Size); AStringStream.Seek(0, soFromBeginning); AStrings.Text := AStringStream.DataString; ReadFromStrings(AStrings, AData); finally AStringStream.Free; AStrings.Free; end; end; procedure TsCustomSpreadReader.ReadFromStrings(AStrings: TStrings; AData: TsWorkbook); begin Unused(AStrings, AData); raise Exception.Create(lpUnsupportedReadFormat); end; { TsCustomSpreadWriter } constructor TsCustomSpreadWriter.Create(AWorkbook: TsWorkbook); begin inherited Create; FWorkbook := AWorkbook; CreateNumFormatList; // FNumFormatList.FWorkbook := AWorkbook; end; destructor TsCustomSpreadWriter.Destroy; begin FNumFormatList.Free; inherited Destroy; end; {@@ Checks if the style of a cell is in the list of manually added FFormattingStyles and returns the index or -1 if it isn't } function TsCustomSpreadWriter.FindFormattingInList(AFormat: PCell): Integer; var i, n: Integer; b: TsCellBorder; equ: Boolean; begin Result := -1; n := Length(FFormattingStyles); for i := n - 1 downto 0 do begin if (FFormattingStyles[i].UsedFormattingFields <> AFormat^.UsedFormattingFields) then Continue; if uffHorAlign in AFormat^.UsedFormattingFields then if (FFormattingStyles[i].HorAlignment <> AFormat^.HorAlignment) then Continue; if uffVertAlign in AFormat^.UsedFormattingFields then if (FFormattingStyles[i].VertAlignment <> AFormat^.VertAlignment) then Continue; if uffTextRotation in AFormat^.UsedFormattingFields then if (FFormattingStyles[i].TextRotation <> AFormat^.TextRotation) then Continue; if uffBorder in AFormat^.UsedFormattingFields then begin if (FFormattingStyles[i].Border <> AFormat^.Border) then Continue; equ := true; for b in TsCellBorder do begin if FFormattingStyles[i].BorderStyles[b].LineStyle <> AFormat^.BorderStyles[b].LineStyle then begin equ := false; Break; end; if FFormattingStyles[i].BorderStyles[b].Color <> AFormat^.BorderStyles[b].Color then begin equ := false; Break; end; end; if not equ then Continue; end; if uffBackgroundColor in AFormat^.UsedFormattingFields then if (FFormattingStyles[i].BackgroundColor <> AFormat^.BackgroundColor) then Continue; if uffNumberFormat in AFormat^.UsedFormattingFields then begin if (FFormattingStyles[i].NumberFormat <> AFormat^.NumberFormat) then Continue; if (FFormattingStyles[i].NumberFormatStr <> AFormat^.NumberFormatStr) then Continue; end; if uffFont in AFormat^.UsedFormattingFields then if (FFormattingStyles[i].FontIndex <> AFormat^.FontIndex) then Continue; // If we arrived here it means that the styles match Exit(i); end; end; { If formatting features of a cell are not supported by the destination file format of the writer, here is the place to apply replacements. Must be overridden by descendants. See BIFF2 } procedure TsCustomSpreadWriter.FixFormat(ACell: PCell); begin Unused(ACell); // to be overridden end; { Each descendent should define its own default formats, if any. Always add the normal, unformatted style first to speed things up. } procedure TsCustomSpreadWriter.AddDefaultFormats(); begin SetLength(FFormattingStyles, 0); NextXFIndex := 0; end; procedure TsCustomSpreadWriter.CreateNumFormatList; begin { The format list needs to be created by descendants who know about the special requirements of the file format. } end; procedure TsCustomSpreadWriter.ListAllFormattingStylesCallback(ACell: PCell; AStream: TStream); var Len: Integer; begin Unused(AStream); FixFormat(ACell); if ACell^.UsedFormattingFields = [] then Exit; if FindFormattingInList(ACell) <> -1 then Exit; Len := Length(FFormattingStyles); SetLength(FFormattingStyles, Len+1); FFormattingStyles[Len] := ACell^; // We store the index of the XF record that will be assigned to this style in // the "row" of the style. Will be needed when writing the XF record. FFormattingStyles[Len].Row := NextXFIndex; Inc(NextXFIndex); end; procedure TsCustomSpreadWriter.ListAllFormattingStyles; var i: Integer; begin SetLength(FFormattingStyles, 0); // Add default styles which are required to be there by the destination file AddDefaultFormats(); // Iterate through all cells and collect the individual styles for i := 0 to Workbook.GetWorksheetCount - 1 do IterateThroughCells(nil, Workbook.GetWorksheetByIndex(i).Cells, ListAllFormattingStylesCallback); end; {@@ Adds the number format of the given cell to the NumFormatList, but only if it does not yet exist in the list. } procedure TsCustomSpreadWriter.ListAllNumFormatsCallback(ACell: PCell; AStream: TStream); var fmt: string; nf: TsNumberFormat; begin Unused(AStream); if ACell^.NumberFormat = nfGeneral then exit; // The builtin format list is in fpc dialect. fmt := ACell^.NumberFormatStr; nf := ACell^.NumberFormat; // Seek the format string in the current number format list. // If not found add the format to the list. if FNumFormatList.Find(nf, fmt) = -1 then FNumFormatList.AddFormat(fmt, nf); end; {@@ Iterates through all cells and collects the number formats in FNumFormatList (without duplicates). The index of the list item is needed for the field FormatIndex of the XF record. At the time when the method is called the formats are still in fpc dialect. } procedure TsCustomSpreadWriter.ListAllNumFormats; var i: Integer; begin for i:=0 to Workbook.GetWorksheetCount-1 do IterateThroughCells(nil, Workbook.GetWorksheetByIndex(i).Cells, ListAllNumFormatsCallback); NumFormatList.Sort; end; {@@ Expands a formula, separating it in it's constituent parts, so that it is already partially parsed and it is easier to convert it into the format supported by the writer module } function TsCustomSpreadWriter.ExpandFormula(AFormula: TsFormula): TsExpandedFormula; var StrPos: Integer; ResPos: Integer; begin ResPos := -1; SetLength(Result, 0); // The formula needs to start with a "=" character. if AFormula.FormulaStr[1] <> '=' then raise Exception.Create('Formula doesn''t start with ='); StrPos := 2; while Length(AFormula.FormulaStr) <= StrPos do begin // Checks for cell with the format [Letter][Number] { if (AFormula.FormulaStr[StrPos] in [a..zA..Z]) and (AFormula.FormulaStr[StrPos + 1] in [0..9]) then begin Inc(ResPos); SetLength(Result, ResPos + 1); Result[ResPos].ElementKind := fekCell; // Result[ResPos].Col1 := fekCell; Result[ResPos].Row1 := AFormula.FormulaStr[StrPos + 1]; Inc(StrPos); end // Checks for arithmetical operations else} if AFormula.FormulaStr[StrPos] = '+' then begin Inc(ResPos); SetLength(Result, ResPos + 1); Result[ResPos].ElementKind := fekAdd; end; Inc(StrPos); end; end; {@@ Helper function for the spreadsheet writers. @see TsCustomSpreadWriter.WriteCellsToStream } procedure TsCustomSpreadWriter.WriteCellCallback(ACell: PCell; AStream: TStream); begin case ACell.ContentType of cctEmpty: WriteBlank(AStream, ACell^.Row, ACell^.Col, ACell); cctDateTime: WriteDateTime(AStream, ACell^.Row, ACell^.Col, ACell^.DateTimeValue, ACell); cctNumber: WriteNumber(AStream, ACell^.Row, ACell^.Col, ACell^.NumberValue, ACell); cctUTF8String: WriteLabel(AStream, ACell^.Row, ACell^.Col, ACell^.UTF8StringValue, ACell); cctFormula: WriteFormula(AStream, ACell^.Row, ACell^.Col, ACell^.FormulaValue, ACell); cctRPNFormula: WriteRPNFormula(AStream, ACell^.Row, ACell^.Col, ACell^.RPNFormulaValue, ACell); end; end; {@@ Helper function for the spreadsheet writers. Iterates all cells on a list, calling the appropriate write method for them. @param AStream The output stream. @param ACells List of cells to be writeen } procedure TsCustomSpreadWriter.WriteCellsToStream(AStream: TStream; ACells: TAVLTree); begin IterateThroughCells(AStream, ACells, WriteCellCallback); end; {@@ A generic method to iterate through all cells in a worksheet and call a callback routine for each cell. @param AStream The output stream, passed to the callback routine. @param ACells List of cells to be iterated @param ACallback The callback routine } procedure TsCustomSpreadWriter.IterateThroughCells(AStream: TStream; ACells: TAVLTree; ACallback: TCellsCallback); var AVLNode: TAVLTreeNode; begin AVLNode := ACells.FindLowest; While Assigned(AVLNode) do begin ACallback(PCell(AVLNode.Data), AStream); AVLNode := ACells.FindSuccessor(AVLNode); end; end; {@@ Default file writting method. Opens the file and calls WriteToStream The workbook written is the one specified in the constructor of the writer. @param AFileName The output file name. If the file already exists it will be replaced. @see TsWorkbook } procedure TsCustomSpreadWriter.WriteToFile(const AFileName: string; const AOverwriteExisting: Boolean = False); var OutputFile: TFileStream; lMode: Word; begin if AOverwriteExisting then lMode := fmCreate or fmOpenWrite else lMode := fmCreate; OutputFile := TFileStream.Create(AFileName, lMode); try WriteToStream(OutputFile); finally OutputFile.Free; end; end; {@@ This routine should be overriden in descendent classes. } procedure TsCustomSpreadWriter.WriteToStream(AStream: TStream); var lStringList: TStringList; begin lStringList := TStringList.Create; try WriteToStrings(lStringList); lStringList.SaveToStream(AStream); finally lStringList.Free; end; end; procedure TsCustomSpreadWriter.WriteToStrings(AStrings: TStrings); begin Unused(AStrings); raise Exception.Create(lpUnsupportedWriteFormat); end; procedure TsCustomSpreadWriter.WriteFormula(AStream: TStream; const ARow, ACol: Cardinal; const AFormula: TsFormula; ACell: PCell); begin Unused(AStream, ARow, ACol); Unused(AFormula, ACell); // Silently dump the formula; child classes should implement their own support end; procedure TsCustomSpreadWriter.WriteRPNFormula(AStream: TStream; const ARow, ACol: Cardinal; const AFormula: TsRPNFormula; ACell: PCell); begin Unused(AStream, ARow, ACol); Unused(AFormula, ACell); // Silently dump the formula; child classes should implement their own support end; { Simplified creation of RPN formulas } {@@ Creates a pointer to a new RPN item. This represents an element in the array of token of an RPN formula. @return Pointer the the RPN item } function NewRPNItem: PRPNItem; begin Result := GetMem(SizeOf(TRPNItem)); FillChar(Result^.FE, SizeOf(Result^.FE), 0); Result^.FE.StringValue := ''; end; {@@ Destroys an RPN item } procedure DisposeRPNItem(AItem: PRPNItem); begin if AItem <> nil then FreeMem(AItem, SizeOf(TRPNItem)); end; {@@ Creates a boolean value entry in the RPN array. @param AValue Boolean value to be stored in the RPN item @next ANext Pointer to the next RPN item in the list } function RPNBool(AValue: Boolean; ANext: PRPNItem): PRPNItem; begin Result := NewRPNItem; Result^.FE.ElementKind := fekBool; if AValue then Result^.FE.DoubleValue := 1.0 else Result^.FE.DoubleValue := 0.0; Result^.Next := ANext; end; {@@ Creates an entry in the RPN array for a cell value, specifed by its address, e.g. 'A1'. Takes care of absolute and relative cell addresses. @param ACellAddress Adress of the cell given in Excel A1 notation @param ANext Pointer to the next RPN item in the list } function RPNCellValue(ACellAddress: String; ANext: PRPNItem): PRPNItem; var r,c: Integer; flags: TsRelFlags; begin if not ParseCellString(ACellAddress, r, c, flags) then raise Exception.CreateFmt('"%s" is not a valid cell address.', [ACellAddress]); Result := RPNCellValue(r,c, flags, ANext); end; {@@ Creates an entry in the RPN array for a cell value, specifed by its row and column index and a flag containing information on relative addresses. @param ARow Row index of the cell @param ACol Column index of the cell @param AFlags Flags specifying absolute or relative cell addresses @param ANext Pointer to the next RPN item in the list } function RPNCellValue(ARow, ACol: Integer; AFlags: TsRelFlags; ANext: PRPNItem): PRPNItem; begin Result := NewRPNItem; Result^.FE.ElementKind := fekCell; Result^.FE.Row := ARow; Result^.FE.Col := ACol; Result^.FE.RelFlags := AFlags; Result^.Next := ANext; end; {@@ Creates an entry in the RPN array for a cell reference, specifed by its address, e.g. 'A1'. Takes care of absolute and relative cell addresses. "Cell reference" means that all properties of the cell can be handled. Note that most Excel formulas with cells require the cell value only (--> RPNCellValue) @param ACellAddress Adress of the cell given in Excel A1 notation @param ANext Pointer to the next RPN item in the list } function RPNCellRef(ACellAddress: String; ANext: PRPNItem): PRPNItem; var r,c: Integer; flags: TsRelFlags; begin if not ParseCellString(ACellAddress, r, c, flags) then raise Exception.CreateFmt(lpNoValidCellAddress, [ACellAddress]); Result := RPNCellRef(r,c, flags, ANext); end; {@@ Creates an entry in the RPN array for a cell reference, specifed by its row and column index and flags containing information on relative addresses. "Cell reference" means that all properties of the cell can be handled. Note that most Excel formulas with cells require the cell value only (--> RPNCellValue) @param ARow Row index of the cell @param ACol Column index of the cell @param AFlags Flags specifying absolute or relative cell addresses @param ANext Pointer to the next RPN item in the list } function RPNCellRef(ARow, ACol: Integer; AFlags: TsRelFlags; ANext: PRPNItem): PRPNItem; begin Result := NewRPNItem; Result^.FE.ElementKind := fekCellRef; Result^.FE.Row := ARow; Result^.FE.Col := ACol; Result^.FE.RelFlags := AFlags; Result^.Next := ANext; end; {@@ Creates an entry in the RPN array for a range of cells, specified by an Excel-style address, e.g. A1:G5. As in Excel, use a $ sign to indicate absolute addresses. @param ACellRangeAddress Adress of the cell range given in Excel notation, such as A1:G5 @param ANext Pointer to the next RPN item in the list } function RPNCellRange(ACellRangeAddress: String; ANext: PRPNItem): PRPNItem; var r1,c1, r2,c2: Integer; flags: TsRelFlags; begin if not ParseCellRangeString(ACellRangeAddress, r1,c1, r2,c2, flags) then raise Exception.CreateFmt(lpNoValidCellRangeAddress, [ACellRangeAddress]); Result := RPNCellRange(r1,c1, r2,c2, flags, ANext); end; {@@ Creates an entry in the RPN array for a range of cells, specified by the row/column indexes of the top/left and bottom/right corners of the block. The flags indicate relative indexes. @param ARow Row index of the top/left cell @param ACol Column index of the top/left cell @param ARow2 Row index of the bottom/right cell @param ACol2 Column index of the bottom/right cell @param AFlags Flags specifying absolute or relative cell addresses @param ANext Pointer to the next RPN item in the list } function RPNCellRange(ARow, ACol, ARow2, ACol2: Integer; AFlags: TsRelFlags; ANext: PRPNItem): PRPNItem; begin Result := NewRPNItem; Result^.FE.ElementKind := fekCellRange; Result^.FE.Row := ARow; Result^.FE.Col := ACol; Result^.FE.Row2 := ARow2; Result^.FE.Col2 := ACol2; Result^.FE.RelFlags := AFlags; Result^.Next := ANext; end; {@@ Creates an entry in the RPN array with an error value. @param AErrCode Error code to be inserted (see TsErrorValue @param ANext Pointer to the next RPN item in the list @see TsErrorValue } function RPNErr(AErrCode: Byte; ANext: PRPNItem): PRPNItem; begin Result := NewRPNItem; Result^.FE.ElementKind := fekErr; Result^.FE.IntValue := AErrCode; Result^.Next := ANext; end; {@@ Creates an entry in the RPN array for a 2-byte unsigned integer @param AValue Integer value to be inserted into the formula @param ANext Pointer to the next RPN item in the list } function RPNInteger(AValue: Word; ANext: PRPNItem): PRPNItem; begin Result := NewRPNItem; Result^.FE.ElementKind := fekInteger; Result^.FE.IntValue := AValue; Result^.Next := ANext; end; {@@ Creates an entry in the RPN array for a missing argument in of function call. Use this in a formula to indicate a missing argument @param ANext Pointer to the next RPN item in the list. } function RPNMissingArg(ANext: PRPNItem): PRPNItem; begin Result := NewRPNItem; Result^.FE.ElementKind := fekMissingArg; Result^.Next := ANext; end; {@@ Creates an entry in the RPN array for a number. Integers and floating-point values can be used likewise. @param AValue Number value to be inserted into the formula @param ANext Pointer to the next RPN item in the list } function RPNNumber(AValue: Double; ANext: PRPNItem): PRPNItem; begin Result := NewRPNItem; Result^.FE.ElementKind := fekNum; Result^.FE.DoubleValue := AValue; Result^.Next := ANext; end; {@@ Creates an entry in the RPN array which puts the current operator in parenthesis. For display purposes only, does not affect calculation. @param ANext Pointer to the next RPN item in the list } function RPNParenthesis(ANext: PRPNItem): PRPNItem; begin Result := NewRPNItem; Result^.FE.ElementKind := fekParen; Result^.Next := ANext; end; {@@ Creates an entry in the RPN array for a string. @param AValue String to be inserted into the formula @param ANext Pointer to the next RPN item in the list } function RPNString(AValue: String; ANext: PRPNItem): PRPNItem; begin Result := NewRPNItem; Result^.FE.ElementKind := fekString; Result^.FE.StringValue := AValue; Result^.Next := ANext; end; {@@ Creates an entry in the RPN array for an Excel function or operation specified by its TokenID (--> TFEKind). Note that array elements for all needed parameters must have been created before. @param AToken Formula element indicating the function to be executed, see the TsFEKind enumeration for possible values. @param ANext Pointer to the next RPN item in the list @see TsFEKind } function RPNFunc(AToken: TFEKind; ANext: PRPNItem): PRPNItem; begin if FEProps[AToken].MinParams <> FEProps[AToken].MaxParams then raise Exception.CreateFmt(lpSpecifyNumberOfParams, [FEProps[AToken].Symbol]); Result := RPNFunc(AToken, FEProps[AToken].MinParams, ANext); end; {@@ Creates an entry in the RPN array for an Excel function or operation specified by its TokenID (--> TFEKind). Specify the number of parameters used. They must have been created before. @param AToken Formula element indicating the function to be executed, see the TsFEKind enumeration for possible values. @param ANumParams Number of arguments used in the formula @param ANext Pointer to the next RPN item in the list @see TsFEKind } function RPNFunc(AToken: TFEKind; ANumParams: Byte; ANext: PRPNItem): PRPNItem; begin if ord(AToken) < ord(fekAdd) then raise Exception.Create('No basic tokens allowed here.'); if (ANumParams < FEProps[AToken].MinParams) or (ANumParams > FEProps[AToken].MaxParams) then raise Exception.CreateFmt(lpIncorrectParamCount, [ FEProps[AToken].Symbol, FEProps[AToken].MinParams, FEProps[AToken].MaxParams ]); Result := NewRPNItem; Result^.FE.ElementKind := AToken; Result^.FE.ParamsNum := ANumParams; Result^.Next := ANext; end; {@@ Returns if the function defined by the token requires a fixed number of parameter. @param AElementKind Identifier of the formula function considered } function FixedParamCount(AElementKind: TFEKind): Boolean; begin Result := (FEProps[AElementKind].MinParams = FEProps[AElementKind].MaxParams) and (FEProps[AElementKind].MinParams >= 0); end; {@@ Creates an RPN formula by a single call using nested RPN items. For each formula element, use one of the RPNxxxx functions implemented here. They are designed to be nested into each other. Terminate the chain by using nil. @param AItem Pointer to the first RPN item representing the formula. Each item contains a pointer to the next item in the list. The list is terminated by nil. @example The RPN formula for the string expression "$A1+2" can be created as follows:
      var
        f: TsRPNFormula;
      begin
        f := CreateRPNFormula(
          RPNCellValue('$A1',
          RPNNumber(2,
          RPNFunc(fekAdd,
          nil))));
    
} function CreateRPNFormula(AItem: PRPNItem): TsRPNFormula; var item: PRPNItem; nextitem: PRPNItem; n: Integer; begin // Determine count of RPN elements n := 0; item := AItem; while item <> nil do begin inc(n); item := item^.Next; end; // Set array length of TsRPNFormula result SetLength(Result, n); // Copy FormulaElements to result and free temporary RPNItems item := AItem; n := 0; while item <> nil do begin nextitem := item^.Next; Result[n] := item^.FE; inc(n); DisposeRPNItem(item); item := nextitem; end; end; {@@ Destroys the RPN formula starting with the given RPN item. @param AItem Pointer to the first RPN items representing the formula. Each item contains a pointer to the next item in the list. The list is terminated by nil. } procedure DestroyRPNFormula(AItem: PRPNItem); var nextitem: PRPNItem; begin while AItem <> nil do begin nextitem := AItem^.Next; DisposeRPNItem(AItem); AItem := nextitem; end; end; initialization MakeLEPalette(@DEFAULT_PALETTE, Length(DEFAULT_PALETTE)); finalization SetLength(GsSpreadFormats, 0); end. { Strategy for handling of number formats: Problem: For number formats, fpspreadsheet uses a syntax which is slightly different from the syntax that Excel uses in the xls files. Moreover, the file syntax can be different from file type to file type (biff2, for example, allows only a few predefined formats, while the number of allowed formats is unlimited (?) for biff8. Number format handling in fpspreadsheet is implemented with the following concept in mind: - Formats written into TsWorksheet cells always follow the fpspreadsheet syntax. - For writing, the writer creates a TsNumFormatList which stores all formats in file syntax. - The built-in formats of the file types are coded in the fpc syntax. - The method "ConvertBeforeWriting" converts the cell formats from the fpspreadsheet to the file syntax. - For reading, the reader creates another TsNumFormatList. - The built-in formats of the file types are coded again in fpc syntax. - After reading, the formats are converted to fpc syntax by means of "ConvertAfterReading". - Format conversion is done internally by means of the TsNumFormatParser. }