
git-svn-id: https://svn.code.sf.net/p/lazarus-ccr/svn@362 8e941d3f-bd1b-0410-a28a-d453659cc2b4
408 lines
17 KiB
ObjectPascal
Executable File
408 lines
17 KiB
ObjectPascal
Executable File
{
|
|
fpsopendocument.pas
|
|
|
|
Writes an OpenDocument 1.0 Spreadsheet document
|
|
|
|
An OpenDocument document is a compressed ZIP file with the following files inside:
|
|
|
|
content.xml
|
|
meta.xml
|
|
settings.xml
|
|
styles.xml
|
|
META-INF\manifest.xml
|
|
|
|
Specifications obtained from:
|
|
|
|
write url here
|
|
|
|
AUTHORS: Felipe Monteiro de Carvalho
|
|
|
|
IMPORTANT: This writer doesn't work yet!!! This is just initial code.
|
|
}
|
|
unit fpsopendocument;
|
|
|
|
{$ifdef fpc}
|
|
{$mode delphi}
|
|
{$endif}
|
|
|
|
interface
|
|
|
|
uses
|
|
Classes, SysUtils, {zipper,}
|
|
fpspreadsheet;
|
|
|
|
type
|
|
|
|
{ TsSpreadOpenDocWriter }
|
|
|
|
TsSpreadOpenDocWriter = class(TsCustomSpreadWriter)
|
|
protected
|
|
// FZip: TZipper;
|
|
FMetaInfManifest: string;
|
|
FMeta, FSettings, FStyles: string;
|
|
FContent: string;
|
|
public
|
|
{ General writing methods }
|
|
procedure WriteStringToFile(AFileName, AString: string);
|
|
procedure WriteToFile(AFileName: string; AData: TsWorkbook); override;
|
|
procedure WriteToStream(AStream: TStream; AData: TsWorkbook); override;
|
|
{ Record writing methods }
|
|
procedure WriteLabel(AStream: TStream; const ARow, ACol: Word; const AValue: string); override;
|
|
procedure WriteNumber(AStream: TStream; const ARow, ACol: Cardinal; const AValue: double); override;
|
|
end;
|
|
|
|
implementation
|
|
|
|
const
|
|
{ OpenDocument general XML constants }
|
|
XML_HEADER = '<?xml version="1.0" encoding="utf-8" ?>';
|
|
|
|
{ OpenDocument Directory structure constants }
|
|
OOXML_PATH_CONTENT = 'content.xml';
|
|
OOXML_PATH_META = 'meta.xml';
|
|
OOXML_PATH_SETTINGS = 'settings.xml';
|
|
OOXML_PATH_STYLES = 'styles.xml';
|
|
OPENDOC_PATH_METAINF = 'META-INF\';
|
|
OPENDOC_PATH_METAINF_MANIFEST = 'META-INF\manifest.xml';
|
|
|
|
{ OpenDocument schemas constants }
|
|
SCHEMAS_XMLNS_OFFICE = 'urn:oasis:names:tc:opendocument:xmlns:office:1.0';
|
|
SCHEMAS_XMLNS_DCTERMS = 'http://purl.org/dc/terms/';
|
|
SCHEMAS_XMLNS_META = 'urn:oasis:names:tc:opendocument:xmlns:meta:1.0';
|
|
SCHEMAS_XMLNS = 'http://schemas.openxmlformats.org/officeDocument/2006/extended-properties';
|
|
SCHEMAS_XMLNS_CONFIG = 'urn:oasis:names:tc:opendocument:xmlns:config:1.0';
|
|
SCHEMAS_XMLNS_OOO = 'http://openoffice.org/2004/office';
|
|
SCHEMAS_XMLNS_MANIFEST = 'urn:oasis:names:tc:opendocument:xmlns:manifest:1.0';
|
|
SCHEMAS_XMLNS_FO = 'urn:oasis:names:tc:opendocument:xmlns:xsl-fo-compatible:1.0';
|
|
SCHEMAS_XMLNS_STYLE = 'urn:oasis:names:tc:opendocument:xmlns:style:1.0';
|
|
SCHEMAS_XMLNS_SVG = 'urn:oasis:names:tc:opendocument:xmlns:svg-compatible:1.0';
|
|
SCHEMAS_XMLNS_TABLE = 'urn:oasis:names:tc:opendocument:xmlns:table:1.0';
|
|
SCHEMAS_XMLNS_TEXT = 'urn:oasis:names:tc:opendocument:xmlns:text:1.0';
|
|
SCHEMAS_XMLNS_V = 'urn:schemas-microsoft-com:vml';
|
|
SCHEMAS_XMLNS_NUMBER = 'urn:oasis:names:tc:opendocument:xmlns:datastyle:1.0';
|
|
SCHEMAS_XMLNS_CHART = 'urn:oasis:names:tc:opendocument:xmlns:chart:1.0';
|
|
SCHEMAS_XMLNS_DR3D = 'urn:oasis:names:tc:opendocument:xmlns:dr3d:1.0';
|
|
SCHEMAS_XMLNS_MATH = 'http://www.w3.org/1998/Math/MathML';
|
|
SCHEMAS_XMLNS_FORM = 'urn:oasis:names:tc:opendocument:xmlns:form:1.0';
|
|
SCHEMAS_XMLNS_SCRIPT = 'urn:oasis:names:tc:opendocument:xmlns:script:1.0';
|
|
SCHEMAS_XMLNS_OOOW = 'http://openoffice.org/2004/writer';
|
|
SCHEMAS_XMLNS_OOOC = 'http://openoffice.org/2004/calc';
|
|
SCHEMAS_XMLNS_DOM = 'http://www.w3.org/2001/xml-events';
|
|
SCHEMAS_XMLNS_XFORMS = 'http://www.w3.org/2002/xforms';
|
|
SCHEMAS_XMLNS_XSD = 'http://www.w3.org/2001/XMLSchema';
|
|
SCHEMAS_XMLNS_XSI = 'http://www.w3.org/2001/XMLSchema-instance';
|
|
|
|
{ TsSpreadOpenDocWriter }
|
|
|
|
{*******************************************************************
|
|
* TsSpreadOOXMLWriter.WriteStringToFile ()
|
|
*
|
|
* DESCRIPTION: Writes a string to a file. Helper convenience method.
|
|
*
|
|
*******************************************************************}
|
|
procedure TsSpreadOpenDocWriter.WriteStringToFile(AFileName, AString: string);
|
|
var
|
|
TheStream : TFileStream;
|
|
S : String;
|
|
begin
|
|
TheStream := TFileStream.Create(AFileName, fmCreate);
|
|
S:=AString;
|
|
TheStream.WriteBuffer(Pointer(S)^,Length(S));
|
|
TheStream.Free;
|
|
end;
|
|
|
|
{*******************************************************************
|
|
* TsSpreadOOXMLWriter.WriteToFile ()
|
|
*
|
|
* DESCRIPTION: Writes an OOXML document to the disc
|
|
*
|
|
*******************************************************************}
|
|
procedure TsSpreadOpenDocWriter.WriteToFile(AFileName: string; AData: TsWorkbook);
|
|
var
|
|
TempDir: string;
|
|
begin
|
|
{FZip := TZipper.Create;
|
|
FZip.ZipFiles(AFileName, x);
|
|
FZip.Free;}
|
|
|
|
WriteToStream(nil, AData);
|
|
|
|
TempDir := IncludeTrailingBackslash(AFileName);
|
|
|
|
{ files on the root path }
|
|
|
|
ForceDirectories(TempDir);
|
|
|
|
WriteStringToFile(TempDir + OOXML_PATH_CONTENT, FContent);
|
|
|
|
WriteStringToFile(TempDir + OOXML_PATH_META, FMeta);
|
|
|
|
WriteStringToFile(TempDir + OOXML_PATH_SETTINGS, FSettings);
|
|
|
|
WriteStringToFile(TempDir + OOXML_PATH_STYLES, FStyles);
|
|
|
|
{ META-INF directory }
|
|
|
|
ForceDirectories(TempDir + OPENDOC_PATH_METAINF);
|
|
|
|
WriteStringToFile(TempDir + OPENDOC_PATH_METAINF_MANIFEST, FMetaInfManifest);
|
|
end;
|
|
|
|
{*******************************************************************
|
|
* TsSpreadOOXMLWriter.WriteToStream ()
|
|
*
|
|
* DESCRIPTION: Writes an Excel 2 file to a stream
|
|
*
|
|
* Excel 2.x files support only one Worksheet per Workbook,
|
|
* so only the first will be written.
|
|
*
|
|
*******************************************************************}
|
|
procedure TsSpreadOpenDocWriter.WriteToStream(AStream: TStream; AData: TsWorkbook);
|
|
begin
|
|
// WriteCellsToStream(AStream, AData.GetFirstWorksheet.FCells);
|
|
|
|
FMetaInfManifest :=
|
|
XML_HEADER + LineEnding +
|
|
'<manifest:manifest xmlns:manifest="' + SCHEMAS_XMLNS_MANIFEST + '">' + LineEnding +
|
|
' <manifest:file-entry manifest:media-type="application/vnd.oasis.opendocument.spreadsheet" manifest:full-path="/" />' + LineEnding +
|
|
' <manifest:file-entry manifest:media-type="text/xml" manifest:full-path="content.xml" />' + LineEnding +
|
|
' <manifest:file-entry manifest:media-type="text/xml" manifest:full-path="styles.xml" />' + LineEnding +
|
|
' <manifest:file-entry manifest:media-type="text/xml" manifest:full-path="meta.xml" />' + LineEnding +
|
|
' <manifest:file-entry manifest:media-type="text/xml" manifest:full-path="settings.xml" />' + LineEnding +
|
|
'</manifest:manifest>';
|
|
|
|
FMeta :=
|
|
XML_HEADER + LineEnding +
|
|
'<office:document-meta xmlns:office="' + SCHEMAS_XMLNS_OFFICE +
|
|
'" xmlns:dcterms="' + SCHEMAS_XMLNS_DCTERMS +
|
|
'" xmlns:meta="' + SCHEMAS_XMLNS_META +
|
|
'" xmlns="' + SCHEMAS_XMLNS +
|
|
'" xmlns:ex="' + SCHEMAS_XMLNS + '">' + LineEnding +
|
|
' <office:meta>' + LineEnding +
|
|
' <meta:generator>FPSpreadsheet Library</meta:generator>' + LineEnding +
|
|
' <meta:document-statistic />' + LineEnding +
|
|
' </office:meta>' + LineEnding +
|
|
'</office:document-meta>';
|
|
|
|
FSettings :=
|
|
XML_HEADER + LineEnding +
|
|
'<office:document-settings xmlns:office="' + SCHEMAS_XMLNS_OFFICE +
|
|
'" xmlns:config="' + SCHEMAS_XMLNS_CONFIG +
|
|
'" xmlns:ooo="' + SCHEMAS_XMLNS_OOO + '">' + LineEnding +
|
|
'<office:settings>' + LineEnding +
|
|
' <config:config-item-set config:name="ooo:view-settings">' + LineEnding +
|
|
' <config:config-item-map-indexed config:name="Views">' + LineEnding +
|
|
' <config:config-item-map-entry>' + LineEnding +
|
|
' <config:config-item config:name="ActiveTable" config:type="string">Tabelle1</config:config-item>' + LineEnding +
|
|
' <config:config-item config:name="ZoomValue" config:type="int">100</config:config-item>' + LineEnding +
|
|
' <config:config-item config:name="PageViewZoomValue" config:type="int">100</config:config-item>' + LineEnding +
|
|
' <config:config-item config:name="ShowPageBreakPreview" config:type="boolean">false</config:config-item>' + LineEnding +
|
|
' <config:config-item config:name="HasColumnRowHeaders" config:type="boolean">true</config:config-item>' + LineEnding +
|
|
' <config:config-item-map-named config:name="Tables">' + LineEnding +
|
|
' <config:config-item-map-entry config:name="Tabelle1">' + LineEnding +
|
|
' <config:config-item config:name="CursorPositionX" config:type="int">3</config:config-item>' + LineEnding +
|
|
' <config:config-item config:name="CursorPositionY" config:type="int">2</config:config-item>' + LineEnding +
|
|
' </config:config-item-map-entry>' + LineEnding +
|
|
' </config:config-item-map-named>' + LineEnding +
|
|
' </config:config-item-map-entry>' + LineEnding +
|
|
' </config:config-item-map-indexed>' + LineEnding +
|
|
' </config:config-item-set>' + LineEnding +
|
|
' </office:settings>' + LineEnding +
|
|
'</office:document-settings>';
|
|
|
|
FStyles :=
|
|
XML_HEADER + LineEnding +
|
|
'<office:document-styles xmlns:office="' + SCHEMAS_XMLNS_OFFICE +
|
|
'" xmlns:fo="' + SCHEMAS_XMLNS_FO +
|
|
'" xmlns:style="' + SCHEMAS_XMLNS_STYLE +
|
|
'" xmlns:svg="' + SCHEMAS_XMLNS_SVG +
|
|
'" xmlns:table="' + SCHEMAS_XMLNS_TABLE +
|
|
'" xmlns:text="' + SCHEMAS_XMLNS_TEXT +
|
|
'" xmlns:v="' + SCHEMAS_XMLNS_V + '">' + LineEnding +
|
|
'<office:font-face-decls>' + LineEnding +
|
|
'<style:font-face style:name="Arial" svg:font-family="Arial" />' + LineEnding +
|
|
'</office:font-face-decls>' + LineEnding +
|
|
'<office:styles>' + LineEnding +
|
|
'<style:style style:name="Default" style:family="table-cell">' + LineEnding +
|
|
'<style:text-properties fo:font-size="10" style:font-name="Arial" />' + LineEnding +
|
|
'</style:style>' + LineEnding +
|
|
'</office:styles>' + LineEnding +
|
|
'<office:automatic-styles>' + LineEnding +
|
|
'<style:page-layout style:name="pm1">' + LineEnding +
|
|
'<style:page-layout-properties fo:margin-top="1.25cm" fo:margin-bottom="1.25cm" fo:margin-left="1.905cm" fo:margin-right="1.905cm" />' + LineEnding +
|
|
'<style:header-style>' + LineEnding +
|
|
'<style:header-footer-properties fo:min-height="0.751cm" fo:margin-left="0cm" fo:margin-right="0cm" fo:margin-bottom="0.25cm" fo:margin-top="0cm" />' + LineEnding +
|
|
'</style:header-style>' + LineEnding +
|
|
'<style:footer-style>' + LineEnding +
|
|
'<style:header-footer-properties fo:min-height="0.751cm" fo:margin-left="0cm" fo:margin-right="0cm" fo:margin-top="0.25cm" fo:margin-bottom="0cm" />' + LineEnding +
|
|
'</style:footer-style>' + LineEnding +
|
|
'</style:page-layout>' + LineEnding +
|
|
'</office:automatic-styles>' + LineEnding +
|
|
'<office:master-styles>' + LineEnding +
|
|
'<style:master-page style:name="Default" style:page-layout-name="pm1">' + LineEnding +
|
|
'<style:header />' + LineEnding +
|
|
'<style:header-left style:display="false" />' + LineEnding +
|
|
'<style:footer />' + LineEnding +
|
|
'<style:footer-left style:display="false" />' + LineEnding +
|
|
'</style:master-page>' + LineEnding +
|
|
'</office:master-styles>' + LineEnding +
|
|
'</office:document-styles>';
|
|
|
|
FContent :=
|
|
XML_HEADER + LineEnding +
|
|
'<office:document-content xmlns:office="' + SCHEMAS_XMLNS_OFFICE +
|
|
'" xmlns:fo="' + SCHEMAS_XMLNS_FO +
|
|
'" xmlns:style="' + SCHEMAS_XMLNS_STYLE +
|
|
'" xmlns:text="' + SCHEMAS_XMLNS_TEXT +
|
|
'" xmlns:table="' + SCHEMAS_XMLNS_TABLE +
|
|
'" xmlns:svg="' + SCHEMAS_XMLNS_SVG +
|
|
'" xmlns:number="' + SCHEMAS_XMLNS_NUMBER +
|
|
'" xmlns:meta="' + SCHEMAS_XMLNS_META +
|
|
'" xmlns:chart="' + SCHEMAS_XMLNS_CHART +
|
|
'" xmlns:dr3d="' + SCHEMAS_XMLNS_DR3D +
|
|
'" xmlns:math="' + SCHEMAS_XMLNS_MATH +
|
|
'" xmlns:form="' + SCHEMAS_XMLNS_FORM +
|
|
'" xmlns:script="' + SCHEMAS_XMLNS_SCRIPT +
|
|
'" xmlns:ooo="' + SCHEMAS_XMLNS_OOO +
|
|
'" xmlns:ooow="' + SCHEMAS_XMLNS_OOOW +
|
|
'" xmlns:oooc="' + SCHEMAS_XMLNS_OOOC +
|
|
'" xmlns:dom="' + SCHEMAS_XMLNS_DOM +
|
|
'" xmlns:xforms="' + SCHEMAS_XMLNS_XFORMS +
|
|
'" xmlns:xsd="' + SCHEMAS_XMLNS_XSD +
|
|
'" xmlns:xsi="' + SCHEMAS_XMLNS_XSI + '">' + LineEnding +
|
|
'<office:scripts />' + LineEnding +
|
|
'<office:font-face-decls>' + LineEnding +
|
|
'<style:font-face style:name="Arial" svg:font-family="Arial" xmlns:v="urn:schemas-microsoft-com:vml" />' + LineEnding +
|
|
'</office:font-face-decls>' + LineEnding +
|
|
'<office:automatic-styles>' + LineEnding +
|
|
'<style:style style:name="ID0EM" style:family="table-column" xmlns:v="urn:schemas-microsoft-com:vml">' + LineEnding +
|
|
'<style:table-column-properties fo:break-before="auto" style:column-width="1.961cm" />' + LineEnding +
|
|
'</style:style>' + LineEnding +
|
|
'<style:style style:name="ID0EM" style:family="table-row" xmlns:v="urn:schemas-microsoft-com:vml">' + LineEnding +
|
|
'<style:table-row-properties fo:break-before="auto" style:row-height="0.45cm" />' + LineEnding +
|
|
'</style:style>' + LineEnding +
|
|
'<style:style style:name="ID1E6B" style:family="table-cell" style:parent-style-name="Default" xmlns:v="urn:schemas-microsoft-com:vml">' + LineEnding +
|
|
'<style:text-properties fo:font-size="10" style:font-name="Arial" />' + LineEnding +
|
|
'</style:style>' + LineEnding +
|
|
'<style:style style:name="ID2EY" style:family="table" style:master-page-name="Default" xmlns:v="urn:schemas-microsoft-com:vml">' + LineEnding +
|
|
'<style:table-properties />' + LineEnding +
|
|
'</style:style>' + LineEnding +
|
|
'<style:style style:name="scenario" style:family="table" style:master-page-name="Default">' + LineEnding +
|
|
'<style:table-properties table:display="false" style:writing-mode="lr-tb" />' + LineEnding +
|
|
'</style:style>' + LineEnding +
|
|
'</office:automatic-styles>' + LineEnding +
|
|
'<office:body>' + LineEnding +
|
|
'<office:spreadsheet>' + LineEnding +
|
|
'<table:table table:name="Tabelle1" table:style-name="ID2EY">' + LineEnding +
|
|
'<table:table-column table:style-name="ID0EM" table:number-columns-repeated="256" table:default-cell-style-name="ID1E6B" xmlns:v="urn:schemas-microsoft-com:vml" />' + LineEnding +
|
|
'<table:table-row table:style-name="ID0EM" xmlns:v="urn:schemas-microsoft-com:vml">' + LineEnding +
|
|
'<table:table-cell office:value-type="float" office:value="1">' + LineEnding +
|
|
'<text:p>1</text:p>' + LineEnding +
|
|
'</table:table-cell>' + LineEnding +
|
|
'<table:table-cell office:value-type="float" office:value="2">' + LineEnding +
|
|
'<text:p>2</text:p>' + LineEnding +
|
|
'</table:table-cell>' + LineEnding +
|
|
'<table:table-cell office:value-type="float" office:value="3">' + LineEnding +
|
|
'<text:p>3</text:p>' + LineEnding +
|
|
'</table:table-cell>' + LineEnding +
|
|
'<table:table-cell office:value-type="float" office:value="4">' + LineEnding +
|
|
'<text:p>4</text:p>' + LineEnding +
|
|
'</table:table-cell>' + LineEnding +
|
|
'<table:table-cell table:number-columns-repeated="252" />' + LineEnding +
|
|
'</table:table-row>' + LineEnding +
|
|
'<table:table-row table:style-name="ID0EM" xmlns:v="urn:schemas-microsoft-com:vml">' + LineEnding +
|
|
'<table:table-cell office:value-type="string">' + LineEnding +
|
|
'<text:p>First</text:p>' + LineEnding +
|
|
'</table:table-cell>' + LineEnding +
|
|
'<table:table-cell office:value-type="string">' + LineEnding +
|
|
'<text:p>Second</text:p>' + LineEnding +
|
|
'</table:table-cell>' + LineEnding +
|
|
'<table:table-cell office:value-type="string">' + LineEnding +
|
|
'<text:p>Third</text:p>' + LineEnding +
|
|
'</table:table-cell>' + LineEnding +
|
|
'<table:table-cell office:value-type="string">' + LineEnding +
|
|
'<text:p>Fourth</text:p>' + LineEnding +
|
|
'</table:table-cell>' + LineEnding +
|
|
'<table:table-cell table:number-columns-repeated="252" />' + LineEnding +
|
|
'</table:table-row>' + LineEnding +
|
|
'</table:table>' + LineEnding +
|
|
'<table:database-ranges />' + LineEnding +
|
|
'</office:spreadsheet>' + LineEnding +
|
|
'</office:body>' + LineEnding +
|
|
'</office:document-content>';
|
|
end;
|
|
|
|
{*******************************************************************
|
|
* TsSpreadOOXMLWriter.WriteLabel ()
|
|
*
|
|
* DESCRIPTION: Writes an Excel 2 LABEL record
|
|
*
|
|
* Writes a string to the sheet
|
|
*
|
|
*******************************************************************}
|
|
procedure TsSpreadOpenDocWriter.WriteLabel(AStream: TStream; const ARow,
|
|
ACol: Word; const AValue: string);
|
|
var
|
|
L: Byte;
|
|
begin
|
|
L := Length(AValue);
|
|
|
|
{ BIFF Record header }
|
|
// AStream.WriteWord(WordToLE(INT_EXCEL_ID_LABEL));
|
|
// AStream.WriteWord(WordToLE(8 + L));
|
|
|
|
{ BIFF Record data }
|
|
// AStream.WriteWord(WordToLE(ARow));
|
|
// AStream.WriteWord(WordToLE(ACol));
|
|
|
|
{ BIFF2 Attributes }
|
|
AStream.WriteByte($0);
|
|
AStream.WriteByte($0);
|
|
AStream.WriteByte($0);
|
|
|
|
{ String with 8-bit size }
|
|
AStream.WriteByte(L);
|
|
AStream.WriteBuffer(AValue[1], L);
|
|
end;
|
|
|
|
{*******************************************************************
|
|
* TsSpreadOOXMLWriter.WriteNumber ()
|
|
*
|
|
* DESCRIPTION: Writes an Excel 2 NUMBER record
|
|
*
|
|
* Writes a number (64-bit IEE 754 floating point) to the sheet
|
|
*
|
|
*******************************************************************}
|
|
procedure TsSpreadOpenDocWriter.WriteNumber(AStream: TStream; const ARow,
|
|
ACol: Cardinal; const AValue: double);
|
|
begin
|
|
{ BIFF Record header }
|
|
// AStream.WriteWord(WordToLE(INT_EXCEL_ID_NUMBER));
|
|
// AStream.WriteWord(WordToLE(15));
|
|
|
|
{ BIFF Record data }
|
|
// AStream.WriteWord(WordToLE(ARow));
|
|
// AStream.WriteWord(WordToLE(ACol));
|
|
|
|
{ BIFF2 Attributes }
|
|
AStream.WriteByte($0);
|
|
AStream.WriteByte($0);
|
|
AStream.WriteByte($0);
|
|
|
|
{ IEE 754 floating-point value }
|
|
AStream.WriteBuffer(AValue, 8);
|
|
end;
|
|
|
|
{*******************************************************************
|
|
* Initialization section
|
|
*
|
|
* Registers this reader / writer on fpSpreadsheet
|
|
*
|
|
*******************************************************************}
|
|
initialization
|
|
|
|
RegisterSpreadFormat(TsCustomSpreadReader, TsSpreadOpenDocWriter, sfOpenDocument);
|
|
|
|
end.
|
|
|