Files
core/ASCOfficeDocFile/DocDocxConverter/StyleSheetMapping.cpp
Alexey.Musinov 9ea51f5678 выпиливание boost'a
git-svn-id: svn://fileserver/activex/AVS/Sources/TeamlabOffice/trunk/ServerComponents@52920 954022d7-b5bf-4e40-9824-e11837661b57
2016-05-20 22:33:45 +03:00

348 lines
12 KiB
C++

#include "stdafx.h"
#include "StyleSheetMapping.h"
namespace DocFileFormat
{
map<std::wstring, std::wstring> StyleSheetMapping::m_mapStyleId;
ASCOfficeCriticalSection StyleSheetMapping::m_mapStyleIdLock;
StyleSheetMapping::StyleSheetMapping( ConversionContext* ctx ) : AbstractOpenXmlMapping( new XmlUtils::CXmlWriter() )
{
_ctx = ctx;
m_document = _ctx->_doc;
}
/*========================================================================================================*/
StyleSheetMapping::~StyleSheetMapping()
{
RELEASEOBJECT( m_pXmlWriter );
}
/*========================================================================================================*/
void StyleSheetMapping::Apply( IVisitable* visited )
{
StyleSheet* sheet = static_cast<StyleSheet*>( visited );
this->_ctx->_docx->RegisterStyleSheet();
//start the document
m_pXmlWriter->WriteNodeBegin( _T( "?xml version=\"1.0\" encoding=\"UTF-8\"?" ) );
m_pXmlWriter->WriteNodeBegin( _T( "w:styles" ), TRUE );
//write namespaces
m_pXmlWriter->WriteAttribute( _T( "xmlns:w" ), OpenXmlNamespaces::WordprocessingML );
m_pXmlWriter->WriteNodeEnd( _T( "" ), TRUE, FALSE );
//document defaults
m_pXmlWriter->WriteNodeBegin( _T( "w:docDefaults" ) );
writeRunDefaults( sheet );
writeParagraphDefaults( sheet );
m_pXmlWriter->WriteNodeEnd( _T( "w:docDefaults" ) );
//write the default styles
if ( sheet->Styles->at( 11 ) == NULL )
{
//NormalTable
writeNormalTableStyle();
}
for ( vector<StyleSheetDescription*>::iterator iter = sheet->Styles->begin(); iter != sheet->Styles->end(); iter++ )
{
if ( *iter != NULL )
{
m_pXmlWriter->WriteNodeBegin( _T( "w:style" ), TRUE );
m_pXmlWriter->WriteAttribute( _T( "w:type" ), FormatUtils::MapValueToWideString( (*iter)->stk, &StyleKindMap[0][0], 5, 10 ).c_str() );
//!!!TODO: There is NO default styles in DOC file. So, we can't choose one of them!!!
/*if ( ( (*iter)->sti != Null ) && ( (*iter)->sti != User ) )
{
//it's a default style
m_pXmlWriter->WriteAttribute( _T( "w:default" ), _T( "1" ) );
}*/
m_pXmlWriter->WriteAttribute( _T( "w:styleId" ), FormatUtils::XmlEncode(MakeStyleId( *iter )).c_str() );
m_pXmlWriter->WriteNodeEnd( _T( "" ), TRUE, FALSE );
// <w:name val="" />
m_pXmlWriter->WriteNodeBegin( _T( "w:name" ), TRUE );
m_pXmlWriter->WriteAttribute( _T( "w:val" ), FormatUtils::XmlEncode(getStyleName( *iter )).c_str() );
m_pXmlWriter->WriteNodeEnd( _T( "" ), TRUE );
// <w:basedOn val="" />
if ( ( (*iter)->istdBase != 4095 ) && ( (*iter)->istdBase < sheet->Styles->size() ) )
{
m_pXmlWriter->WriteNodeBegin( _T( "w:basedOn" ), TRUE );
m_pXmlWriter->WriteAttribute( _T( "w:val" ), FormatUtils::XmlEncode(MakeStyleId( sheet->Styles->at( (*iter)->istdBase ) )).c_str() );
m_pXmlWriter->WriteNodeEnd( _T( "" ), TRUE );
}
// <w:next val="" />
if ( (*iter)->istdNext < sheet->Styles->size() )
{
m_pXmlWriter->WriteNodeBegin( _T( "w:next" ), TRUE );
m_pXmlWriter->WriteAttribute( _T( "w:val" ), FormatUtils::XmlEncode(MakeStyleId( sheet->Styles->at( (*iter)->istdNext ) )).c_str() );
m_pXmlWriter->WriteNodeEnd( _T( "" ), TRUE );
}
// <w:link val="" />
if ( (*iter)->istdLink < sheet->Styles->size() )
{
m_pXmlWriter->WriteNodeBegin( _T( "w:link" ), TRUE );
m_pXmlWriter->WriteAttribute( _T( "w:val" ), FormatUtils::XmlEncode(MakeStyleId( sheet->Styles->at( (*iter)->istdLink ) )).c_str() );
m_pXmlWriter->WriteNodeEnd( _T( "" ), TRUE );
}
// <w:locked/>
if ( (*iter)->fLocked )
{
m_pXmlWriter->WriteNodeBegin( _T( "w:locked" ) );
m_pXmlWriter->WriteNodeEnd( _T( "w:locked" ) );
}
// <w:hidden/>
if ( (*iter)->fHidden )
{
m_pXmlWriter->WriteNodeBegin( _T( "w:hidden" ) );
m_pXmlWriter->WriteNodeEnd( _T( "w:hidden" ) );
}
// <w:semiHidden/>
if ( (*iter)->fSemiHidden )
{
m_pXmlWriter->WriteNodeBegin( _T( "w:semiHidden" ) );
m_pXmlWriter->WriteNodeEnd( _T( "w:semiHidden" ) );
}
//write paragraph properties
if ( (*iter)->papx != NULL )
{
ParagraphPropertiesMapping* ppMappingnew = new ParagraphPropertiesMapping (m_pXmlWriter, _ctx, m_document, NULL, false);
(*iter)->papx->Convert( ppMappingnew );
RELEASEOBJECT( ppMappingnew );
}
//write character properties
if ( (*iter)->chpx != NULL )
{
RevisionData rev;
rev.Type = NoRevision;
CharacterPropertiesMapping* cpMapping = new CharacterPropertiesMapping (m_pXmlWriter, m_document, &rev, (*iter)->papx, true, false);
(*iter)->chpx->Convert( cpMapping );
RELEASEOBJECT( cpMapping );
}
//write table properties
if ( (*iter)->tapx != NULL )
{
vector<short> tableGrid;
TablePropertiesMapping* tpMapping = new TablePropertiesMapping (m_pXmlWriter, sheet, &tableGrid, false);
(*iter)->tapx->Convert( tpMapping );
RELEASEOBJECT( tpMapping );
}
m_pXmlWriter->WriteNodeEnd( _T( "w:style" ) );
}
}
m_pXmlWriter->WriteNodeEnd( _T( "w:styles" ) );
this->_ctx->_docx->StyleSheetXML = wstring( m_pXmlWriter->GetXmlString() );
}
/*========================================================================================================*/
/// Generates a style id for custom style names or returns the build-in identifier for build-in styles.
wstring StyleSheetMapping::MakeStyleId( StyleSheetDescription* std )
{
wstring ret;
if ( std != NULL )
{
if ( ( std->sti != User ) && ( std->sti != Null ) )
{
//use the identifier
if ( std->sti < 159 )
{
ret = wstring( StyleIdentifierMap[std->sti] );
}
else
{
ret = FormatUtils::IntToWideString( std->sti );
}
}
else
{
//if no identifier is set, use the unique id.
map<std::wstring, std::wstring>::const_iterator findResult = m_mapStyleId.find(std->xstzName);
if( findResult != m_mapStyleId.end() )
{
ret = findResult->second;
}
else
{
//genarate new
m_mapStyleIdLock.Enter();
int nIndex = m_mapStyleId.size();
ret = _T("StGen") + FormatUtils::IntToWideString( nIndex );
pair< std::wstring, std::wstring > p( std->xstzName, ret );
m_mapStyleId.insert(p);
m_mapStyleIdLock.Leave();
}
}
}
return ret;
}
/*========================================================================================================*/
void StyleSheetMapping::writeRunDefaults( StyleSheet* sheet )
{
m_pXmlWriter->WriteNodeBegin( _T( "w:rPrDefault" ) );
m_pXmlWriter->WriteNodeBegin( _T( "w:rPr" ) );
//write default fonts
m_pXmlWriter->WriteNodeBegin( _T( "w:rFonts" ), TRUE );
FontFamilyName* ffnAscii = static_cast<FontFamilyName*>( m_document->FontTable->operator [] ( sheet->stshi->rgftcStandardChpStsh[0] ) );
m_pXmlWriter->WriteAttribute( _T( "w:ascii" ), FormatUtils::XmlEncode(ffnAscii->xszFtn).c_str() );
FontFamilyName* ffnAsia = static_cast<FontFamilyName*>( m_document->FontTable->operator [] ( sheet->stshi->rgftcStandardChpStsh[1] ) );
m_pXmlWriter->WriteAttribute( _T( "w:eastAsia" ), FormatUtils::XmlEncode(ffnAsia->xszFtn).c_str() );
FontFamilyName* ffnAnsi = static_cast<FontFamilyName*>( m_document->FontTable->operator [] ( sheet->stshi->rgftcStandardChpStsh[2] ) );
m_pXmlWriter->WriteAttribute( _T( "w:hAnsi" ), FormatUtils::XmlEncode(ffnAnsi->xszFtn).c_str() );
FontFamilyName* ffnComplex = static_cast<FontFamilyName*>( m_document->FontTable->operator [] ( sheet->stshi->rgftcStandardChpStsh[3] ) );
m_pXmlWriter->WriteAttribute( _T( "w:cs" ), FormatUtils::XmlEncode(ffnComplex->xszFtn).c_str() );
m_pXmlWriter->WriteNodeEnd( _T( "" ), TRUE, FALSE );
m_pXmlWriter->WriteNodeEnd( _T( "w:rFonts" ) );
m_pXmlWriter->WriteNodeEnd( _T( "w:rPr" ) );
m_pXmlWriter->WriteNodeEnd( _T( "w:rPrDefault" ) );
}
/*========================================================================================================*/
void StyleSheetMapping::writeParagraphDefaults( StyleSheet* sheet )
{
//if there is no pPrDefault, Word will not used the default paragraph settings.
//writing an empty pPrDefault will cause Word to load the default paragraph settings.
m_pXmlWriter->WriteNodeBegin( _T( "w:pPrDefault" ) );
m_pXmlWriter->WriteNodeEnd( _T( "w:pPrDefault" ) );
}
/*========================================================================================================*/
/// Chooses the correct style name.
/// Word 2007 needs the identifier instead of the stylename for translating it into the UI language.
wstring StyleSheetMapping::getStyleName( StyleSheetDescription* std )
{
wstring ret;
if ( std != NULL )
{
if ( ( std->sti != User ) && ( std->sti != Null ) )
{
//use the identifier
if ( std->sti < 159 )
{
ret = wstring( StyleIdentifierMap[std->sti] );
}
else
{
ret = FormatUtils::IntToWideString( std->sti );
}
}
else
{
//if no identifier is set, use the unique id.
map<std::wstring, std::wstring>::const_iterator findResult = m_mapStyleId.find(std->xstzName);
if( findResult != m_mapStyleId.end() )
{
ret = findResult->second;
}
else
{
//genarate new
m_mapStyleIdLock.Enter();
int nIndex = m_mapStyleId.size();
ret = _T("StGen") + FormatUtils::IntToWideString( nIndex );
pair< std::wstring, std::wstring > p( std->xstzName, ret );
m_mapStyleId.insert(p);
m_mapStyleIdLock.Leave();
}
}
}
return ret;
}
/*========================================================================================================*/
/// Writes the "NormalTable" default style
void StyleSheetMapping::writeNormalTableStyle()
{
m_pXmlWriter->WriteNodeBegin( _T( "w:style" ), TRUE );
m_pXmlWriter->WriteAttribute( _T( "w:type" ), _T( "table" ) );
m_pXmlWriter->WriteAttribute( _T( "w:default" ), _T( "1" ) );
m_pXmlWriter->WriteAttribute( _T( "w:styleId" ), _T( "TableNormal" ) );
m_pXmlWriter->WriteNodeEnd( _T( "" ), TRUE, FALSE );
m_pXmlWriter->WriteNodeBegin( _T( "w:name" ), TRUE );
m_pXmlWriter->WriteAttribute( _T( "w:val" ), _T( "Normal Table" ) );
m_pXmlWriter->WriteNodeEnd( _T( "" ), TRUE );
m_pXmlWriter->WriteNodeBegin( _T( "w:uiPriority" ), TRUE );
m_pXmlWriter->WriteAttribute( _T( "w:val" ), _T( "99" ) );
m_pXmlWriter->WriteNodeEnd( _T( "" ), TRUE );
m_pXmlWriter->WriteNodeBegin( _T( "w:semiHidden" ), TRUE );
m_pXmlWriter->WriteNodeEnd( _T( "" ), TRUE );
m_pXmlWriter->WriteNodeBegin( _T( "w:unhideWhenUsed" ), TRUE );
m_pXmlWriter->WriteNodeEnd( _T( "" ), TRUE );
m_pXmlWriter->WriteNodeBegin( _T( "w:qFormat" ), TRUE );
m_pXmlWriter->WriteNodeEnd( _T( "" ), TRUE );
m_pXmlWriter->WriteNodeBegin( _T( "w:tblPr" ) );
m_pXmlWriter->WriteNodeBegin( _T( "w:tblInd" ), TRUE );
m_pXmlWriter->WriteAttribute( _T( "w:w" ), _T( "0" ) );
m_pXmlWriter->WriteAttribute( _T( "w:type" ), _T( "dxa" ) );
m_pXmlWriter->WriteNodeEnd( _T( "" ), TRUE );
m_pXmlWriter->WriteNodeBegin( _T( "w:tblCellMar" ) );
m_pXmlWriter->WriteNodeBegin( _T( "w:top" ), TRUE );
m_pXmlWriter->WriteAttribute( _T( "w:w" ), _T( "0" ) );
m_pXmlWriter->WriteAttribute( _T( "w:type" ), _T( "dxa" ) );
m_pXmlWriter->WriteNodeEnd( _T( "" ), TRUE );
m_pXmlWriter->WriteNodeBegin( _T( "w:left" ), TRUE );
m_pXmlWriter->WriteAttribute( _T( "w:w" ), _T( "108" ) );
m_pXmlWriter->WriteAttribute( _T( "w:type" ), _T( "dxa" ) );
m_pXmlWriter->WriteNodeEnd( _T( "" ), TRUE );
m_pXmlWriter->WriteNodeBegin( _T( "w:bottom" ), TRUE );
m_pXmlWriter->WriteAttribute( _T( "w:w" ), _T( "0" ) );
m_pXmlWriter->WriteAttribute( _T( "w:type" ), _T( "dxa" ) );
m_pXmlWriter->WriteNodeEnd( _T( "" ), TRUE );
m_pXmlWriter->WriteNodeBegin( _T( "w:right" ), TRUE );
m_pXmlWriter->WriteAttribute( _T( "w:w" ), _T( "108" ) );
m_pXmlWriter->WriteAttribute( _T( "w:type" ), _T( "dxa" ) );
m_pXmlWriter->WriteNodeEnd( _T( "" ), TRUE );
m_pXmlWriter->WriteNodeEnd( _T( "w:tblCellMar" ) );
m_pXmlWriter->WriteNodeEnd( _T( "w:tblPr" ) );
m_pXmlWriter->WriteNodeEnd( _T( "w:style" ) );
}
}