/* -*- Mode: C++; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- */ /* * This file is part of the LibreOffice project. * * This Source Code Form is subject to the terms of the Mozilla Public * License, v. 2.0. If a copy of the MPL was not distributed with this * file, You can obtain one at http://mozilla.org/MPL/2.0/. * * This file incorporates work covered by the following license notice: * * Licensed to the Apache Software Foundation (ASF) under one or more * contributor license agreements. See the NOTICE file distributed * with this work for additional information regarding copyright * ownership. The ASF licenses this file to you under the Apache * License, Version 2.0 (the "License"); you may not use this file * except in compliance with the License. You may obtain a copy of * the License at http://www.apache.org/licenses/LICENSE-2.0 . */ #include #include #include #include "hwpreader.hxx" #include #include #include "fontmap.hxx" #include "formula.h" #include "cspline.h" #include #include #include // #i42367# prevent MS compiler from using system locale for parsing #ifdef _MSC_VER #pragma setlocale("C") #endif // To be shorten source code by realking #define hconv(x) OUString(hstr2ucsstr(x).c_str()) #define ascii(x) OUString::createFromAscii(x) #define rstartEl(x,y) do { if (m_rxDocumentHandler.is()) m_rxDocumentHandler->startElement(x,y); } while(false) #define rendEl(x) do { if (m_rxDocumentHandler.is()) m_rxDocumentHandler->endElement(x); } while(false) #define rchars(x) do { if (m_rxDocumentHandler.is()) m_rxDocumentHandler->characters(x); } while(false) #define padd(x,y,z) pList->addAttribute(x,y,z) #define Double2Str(x) OUString::number((double)(x)) #define WTI(x) ((double)(x) / 1800.) // unit => inch #define WTMM(x) ((double)(x) / 1800. * 25.4) // unit => mm #define WTSM(x) ((int)((x) / 1800. * 2540)) // unit ==> 1/100 mm #define PI 3.14159265358979323846 // xmloff/xmlkyd.hxx #define sXML_CDATA ascii("CDATA") #define STARTP padd( ascii("text:style-name"), ascii("CDATA"), ascii(getPStyleName(((ParaShape &)para->GetParaShape()).index,buf))); \ rstartEl( ascii("text:p"),rList ); \ pList->clear(); \ pstart = true #define STARTT \ curr = para->GetCharShape(n > 0 ? n-1 : 0)->index; \ padd( ascii("text:style-name"), ascii("CDATA") , ascii( getTStyleName(curr, buf) ) ); \ rstartEl( ascii("text:span"),rList ); \ pList->clear(); \ tstart = true #define ENDP \ rendEl(ascii("text:p")); \ pstart = false #define ENDT \ rendEl(ascii("text:span")); \ tstart = false static hchar *field = 0L; static char buf[1024]; namespace { template struct Free { void operator()(T* const ptr) { free(ptr); } }; } struct HwpReaderPrivate { HwpReaderPrivate() { bFirstPara = true; bInBody = false; bInHeader = false; nPnPos = 0; pPn = 0L; } bool bFirstPara; bool bInBody; bool bInHeader; ShowPageNum *pPn; int nPnPos; }; HwpReader::HwpReader() { pList = new AttributeListImpl; rList = (XAttributeList *) pList; d = new HwpReaderPrivate; } HwpReader::~HwpReader() { rList = 0; delete d; } sal_Bool HwpReader::filter(const Sequence< PropertyValue >& rDescriptor) throw(RuntimeException, std::exception) { utl::MediaDescriptor aDescriptor(rDescriptor); aDescriptor.addInputStream(); Reference< XInputStream > xInputStream( aDescriptor[utl::MediaDescriptor::PROP_INPUTSTREAM()], UNO_QUERY_THROW); std::unique_ptr stream(new HStream); Sequence < sal_Int8 > aBuffer; sal_Int32 nRead, nBlock = 32768, nTotal = 0; while( true ) { nRead = xInputStream->readBytes(aBuffer, nBlock); if( nRead == 0 ) break; stream->addData( (const byte *)aBuffer.getConstArray(), nRead ); nTotal += nRead; } if( nTotal == 0 ) return sal_False; if (hwpfile.ReadHwpFile(stream.release())) return sal_False; if (m_rxDocumentHandler.is()) m_rxDocumentHandler->startDocument(); padd(ascii("office:class"), sXML_CDATA, ascii("text")); padd(ascii("office:version"), sXML_CDATA, ascii("0.9")); padd(ascii("xmlns:office"), ascii("CDATA"), ascii("http://openoffice.org/2000/office")); padd(ascii("xmlns:style"), ascii("CDATA"), ascii("http://openoffice.org/2000/style")); padd(ascii("xmlns:text"), ascii("CDATA"), ascii("http://openoffice.org/2000/text")); padd(ascii("xmlns:table"), ascii("CDATA"), ascii("http://openoffice.org/2000/table")); padd(ascii("xmlns:draw"), ascii("CDATA"), ascii("http://openoffice.org/2000/drawing")); padd(ascii("xmlns:fo"), ascii("CDATA"), ascii("http://www.w3.org/1999/XSL/Format")); padd(ascii("xmlns:xlink"), ascii("CDATA"), ascii("http://www.w3.org/1999/xlink")); padd(ascii("xmlns:dc"), ascii("CDATA"), ascii("http://purl.org/dc/elements/1.1/")); padd(ascii("xmlns:meta"), ascii("CDATA"), ascii("http://openoffice.org/2000/meta")); padd(ascii("xmlns:number"), ascii("CDATA"), ascii("http://openoffice.org/2000/datastyle")); padd(ascii("xmlns:svg"), ascii("CDATA"), ascii("http://www.w3.org/2000/svg")); padd(ascii("xmlns:chart"), ascii("CDATA"), ascii("http://openoffice.org/2000/chart")); padd(ascii("xmlns:dr3d"), ascii("CDATA"), ascii("http://openoffice.org/2000/dr3d")); padd(ascii("xmlns:math"), ascii("CDATA"), ascii("http://www.w3.org/1998/Math/MathML")); padd(ascii("xmlns:form"), ascii("CDATA"), ascii("http://openoffice.org/2000/form")); padd(ascii("xmlns:script"), ascii("CDATA"), ascii("http://openoffice.org/2000/script")); rstartEl(ascii("office:document"), rList); pList->clear(); makeMeta(); makeStyles(); makeAutoStyles(); makeMasterStyles(); makeBody(); rendEl(ascii("office:document")); if (m_rxDocumentHandler.is()) m_rxDocumentHandler->endDocument(); return sal_True; } /** * make office:body */ void HwpReader::makeBody() { rstartEl(ascii("office:body"), rList); makeTextDecls(); HWPPara *hwppara = hwpfile.GetFirstPara(); d->bInBody = true; parsePara(hwppara); rendEl(ascii("office:body")); d->bInBody = false; } /** * make text decls */ void HwpReader::makeTextDecls() { rstartEl(ascii("text:sequence-decls"), rList); padd(ascii("text:display-outline-level"), sXML_CDATA, ascii("0")); padd(ascii("text:name"), sXML_CDATA, ascii("Illustration")); rstartEl(ascii("text:sequence-decl"), rList); pList->clear(); rendEl(ascii("text:sequence-decl")); padd(ascii("text:display-outline-level"), sXML_CDATA, ascii("0")); padd(ascii("text:name"), sXML_CDATA, ascii("Table")); rstartEl(ascii("text:sequence-decl"), rList); pList->clear(); rendEl(ascii("text:sequence-decl")); padd(ascii("text:display-outline-level"), sXML_CDATA, ascii("0")); padd(ascii("text:name"), sXML_CDATA, ascii("Text")); rstartEl(ascii("text:sequence-decl"), rList); pList->clear(); rendEl(ascii("text:sequence-decl")); padd(ascii("text:display-outline-level"), sXML_CDATA, ascii("0")); padd(ascii("text:name"), sXML_CDATA, ascii("Drawing")); rstartEl(ascii("text:sequence-decl"), rList); pList->clear(); rendEl(ascii("text:sequence-decl")); rendEl(ascii("text:sequence-decls")); } #define ISNUMBER(x) ( (x) <= 0x39 && (x) >= 0x30 ) /** * make office:meta * Completed */ void HwpReader::makeMeta() { HWPInfo& hwpinfo = hwpfile.GetHWPInfo(); rstartEl(ascii("office:meta"), rList); if (hwpinfo.summary.title[0]) { rstartEl(ascii("dc:title"), rList); rchars((hconv(hwpinfo.summary.title))); rendEl(ascii("dc:title")); } if (hwpinfo.summary.subject[0]) { rstartEl(ascii("dc:subject"), rList); rchars((hconv(hwpinfo.summary.subject))); rendEl(ascii("dc:subject")); } if (hwpinfo.summary.author[0]) { rstartEl(ascii("meta:initial-creator"), rList); rchars((hconv(hwpinfo.summary.author))); rendEl(ascii("meta:initial-creator")); } if (hwpinfo.summary.date[0]) { unsigned short *pDate = hwpinfo.summary.date; int year,month,day,hour,minute; int gab = 0; if( ISNUMBER( pDate[0] ) && ISNUMBER( pDate[1] ) && ISNUMBER( pDate[2] ) && ISNUMBER( pDate[3] )) { year = (pDate[0]-0x30) * 1000 + (pDate[1]-0x30) * 100 + (pDate[2]-0x30) * 10 + (pDate[3]-0x30); } else { year = 0; } if( ISNUMBER( pDate[6] )) { if( ISNUMBER( pDate[7] ) ) month = (pDate[6] - 0x30) * 10 + (pDate[6+ ++gab]-0x30); else month = (pDate[6] - 0x30); } else { month = 0; } if( ISNUMBER( pDate[9 + gab] ) ) { if( ISNUMBER( pDate[10 + gab])) { day = ( pDate[9 + gab] - 0x30 ) * 10 + (pDate[9+ gab + 1]-0x30); ++gab; } else day = (pDate[9+gab]-0x30); } else { day = 0; } if( ISNUMBER( pDate[17 + gab] ) ) { if( ISNUMBER( pDate[18 + gab])) { hour = ( pDate[17 + gab] - 0x30 ) * 10 + (pDate[17+ gab + 1]-0x30); ++gab; } else hour = (pDate[17+gab]-0x30); } else { hour = 0; } if( ISNUMBER( pDate[20 + gab] ) ) { if( ISNUMBER( pDate[21 + gab])) { minute = ( pDate[20 + gab] - 0x30 ) * 10 + (pDate[20+ gab + 1]-0x30); ++gab; } else minute = (pDate[20+gab]-0x30); } else { minute = 0; } sprintf(buf,"%d-%02d-%02dT%02d:%02d:00",year,month,day,hour,minute); rstartEl( ascii("meta:creation-date"), rList ); rchars( ascii(buf)); rendEl( ascii("meta:creation-date") ); } if (hwpinfo.summary.keyword[0][0] || hwpinfo.summary.etc[0][0]) { rstartEl(ascii("meta:keywords"), rList); if (hwpinfo.summary.keyword[0][0]) { rstartEl(ascii("meta:keyword"), rList); rchars((hconv(hwpinfo.summary.keyword[0]))); rendEl(ascii("meta:keyword")); } if (hwpinfo.summary.keyword[1][0]) { rstartEl(ascii("meta:keyword"), rList); rchars((hconv(hwpinfo.summary.keyword[1]))); rendEl(ascii("meta:keyword")); } if (hwpinfo.summary.etc[0][0]) { rstartEl(ascii("meta:keyword"), rList); rchars((hconv(hwpinfo.summary.etc[0]))); rendEl(ascii("meta:keyword")); } if (hwpinfo.summary.etc[1][0]) { rstartEl(ascii("meta:keyword"), rList); rchars((hconv(hwpinfo.summary.etc[1]))); rendEl(ascii("meta:keyword")); } if (hwpinfo.summary.etc[2][0]) { rstartEl(ascii("meta:keyword"), rList); rchars((hconv(hwpinfo.summary.etc[2]))); rendEl(ascii("meta:keyword")); } rendEl(ascii("meta:keywords")); } rendEl(ascii("office:meta")); } static struct { const char *name; bool bMade; } ArrowShape[] = { { "", false }, { "Arrow", false }, { "Line Arrow", false }, { "Square", false } }; static struct { double dots1; double dots2; double distance; } LineStyle[] = { { 0.0, 0.0, 0.0 }, { 0.34, 0., 0.272 }, { 0.17, 0., 0.136}, { 0.612, 0.17, 0.136 }, { 0.85, 0.17, 0.136} }; void HwpReader::makeDrawMiscStyle( HWPDrawingObject *hdo ) { while( hdo ) { if( hdo->child ) makeDrawMiscStyle( hdo->child ); HWPDOProperty *prop = &hdo->property; if( hdo->type == HWPDO_CONTAINER ) { hdo = hdo->next; continue; } if( prop->line_pstyle > 0 && prop->line_pstyle < 5 && prop->line_color <= 0xffffff) { padd( ascii("draw:name"), sXML_CDATA, ascii(Int2Str(hdo->index, "LineType%d", buf))); padd( ascii("draw:style"), sXML_CDATA, ascii("round")); padd( ascii("draw:dots1"), sXML_CDATA, ascii("1")); padd( ascii("draw:dots1-length"), sXML_CDATA, Double2Str( LineStyle[prop->line_pstyle].dots1 * WTMM(prop->line_width) ) + ascii("cm")); if( prop->line_pstyle == 3 ) { padd( ascii("draw:dots2"), sXML_CDATA, ascii("1")); padd( ascii("draw:dots2-length"), sXML_CDATA, Double2Str( LineStyle[prop->line_pstyle].dots2 * WTMM(prop->line_width) ) + ascii("cm")); } else if( prop->line_pstyle == 4 ) { padd( ascii("draw:dots2"), sXML_CDATA, ascii("2")); padd( ascii("draw:dots2-length"), sXML_CDATA, Double2Str( LineStyle[prop->line_pstyle].dots2 * WTMM(prop->line_width)) + ascii("cm")); } padd( ascii("draw:distance"), sXML_CDATA, Double2Str( LineStyle[prop->line_pstyle].distance * WTMM(prop->line_width)) + ascii("cm")); rstartEl( ascii("draw:stroke-dash"), rList); pList->clear(); rendEl( ascii("draw:stroke-dash") ); } if( hdo->type == HWPDO_LINE || hdo->type == HWPDO_ARC || hdo->type == HWPDO_FREEFORM || hdo->type == HWPDO_ADVANCED_ARC ) { if( prop->line_tstyle && !ArrowShape[prop->line_tstyle].bMade ) { ArrowShape[prop->line_tstyle].bMade = true; padd(ascii("draw:name"), sXML_CDATA, ascii(ArrowShape[prop->line_tstyle].name)); if( prop->line_tstyle == 1 ) { padd(ascii("svg:viewBox"), sXML_CDATA, ascii("0 0 20 30")); padd(ascii("svg:d"), sXML_CDATA, ascii("m10 0-10 30h20z")); } else if( prop->line_tstyle == 2 ) { padd(ascii("svg:viewBox"), sXML_CDATA, ascii("0 0 1122 2243")); padd(ascii("svg:d"), sXML_CDATA, ascii("m0 2108v17 17l12 42 30 34 38 21 43 4 29-8 30-21 25-26 13-34 343-1532 339 1520 13 42 29 34 39 21 42 4 42-12 34-30 21-42v-39-12l-4 4-440-1998-9-42-25-39-38-25-43-8-42 8-38 25-26 39-8 42z")); } else if( prop->line_tstyle == 3 ) { padd(ascii("svg:viewBox"), sXML_CDATA, ascii("0 0 30 30")); padd(ascii("svg:d"), sXML_CDATA, ascii("m0 0h30v30h-30z")); } rstartEl(ascii("draw:marker"), rList); pList->clear(); rendEl(ascii("draw:marker")); } if( prop->line_hstyle && !ArrowShape[prop->line_hstyle].bMade) { ArrowShape[prop->line_hstyle].bMade = true; padd(ascii("draw:name"), sXML_CDATA, ascii(ArrowShape[prop->line_hstyle].name)); if( prop->line_hstyle == 1 ) { padd(ascii("svg:viewBox"), sXML_CDATA, ascii("0 0 20 30")); padd(ascii("svg:d"), sXML_CDATA, ascii("m10 0-10 30h20z")); } else if( prop->line_hstyle == 2 ) { padd(ascii("svg:viewBox"), sXML_CDATA, ascii("0 0 1122 2243")); padd(ascii("svg:d"), sXML_CDATA, ascii("m0 2108v17 17l12 42 30 34 38 21 43 4 29-8 30-21 25-26 13-34 343-1532 339 1520 13 42 29 34 39 21 42 4 42-12 34-30 21-42v-39-12l-4 4-440-1998-9-42-25-39-38-25-43-8-42 8-38 25-26 39-8 42z")); } else if( prop->line_hstyle == 3 ) { padd(ascii("svg:viewBox"), sXML_CDATA, ascii("0 0 20 20")); padd(ascii("svg:d"), sXML_CDATA, ascii("m0 0h20v20h-20z")); } rstartEl(ascii("draw:marker"), rList); pList->clear(); rendEl(ascii("draw:marker")); } } if( hdo->type != HWPDO_LINE ) { if( prop->flag >> 18 & 0x01 ) { padd( ascii("draw:name"), sXML_CDATA, ascii(Int2Str(hdo->index, "fillimage%d", buf))); if( !prop->pictype ) { padd( ascii("xlink:href"), sXML_CDATA, hconv(kstr2hstr( (uchar *)urltounix(prop->szPatternFile).c_str()).c_str())); } else { EmPicture *emp = 0L; if ( strlen( prop->szPatternFile ) > 3) emp = hwpfile.GetEmPictureByName(prop->szPatternFile); if( emp ) { char filename[128+17+9]; char dirname[128]; int fd; #ifdef _WIN32 GetTempPath(sizeof(dirname), dirname); sprintf(filename, "%s%s",dirname, emp->name); if( (fd = open( filename , _O_CREAT | _O_WRONLY | _O_BINARY , 0666)) >= 0 ) #else strcpy(dirname, "/tmp/"); sprintf(filename, "%s%s", dirname, emp->name); if( (fd = open( filename , O_CREAT | O_WRONLY , 0666)) >= 0 ) #endif { size_t nWritten = write(fd, emp->data, emp->size); OSL_VERIFY(nWritten == emp->size); close(fd); } #ifdef _WIN32 int j; for(j = 0 ; j < (int)strlen( dirname ) ; j++) { if( dirname[j] == '\\' ) buf[j] = '/'; else buf[j] = dirname[j]; } buf[j] = '\0'; sprintf(filename, "file:///%s%s",buf, emp->name ); #else sprintf(filename, "file://%s%s",dirname, emp->name ); #endif padd( ascii("xlink:href"), sXML_CDATA, ascii(filename)); } else { padd( ascii("xlink:href"), sXML_CDATA, hconv(kstr2hstr( (uchar *)urltounix(prop->szPatternFile).c_str()).c_str())); } } padd( ascii("xlink:type"), sXML_CDATA, ascii("simple")); padd( ascii("xlink:show"), sXML_CDATA, ascii("embed")); padd( ascii("xlink:actuate"), sXML_CDATA, ascii("onLoad")); rstartEl( ascii("draw:fill-image"), rList); pList->clear(); rendEl( ascii("draw:fill-image")); } /* 그라데이션이 존재해도, 비트맵파일이 존재하면, 이것이 우선이다. */ else if( prop->flag >> 16 & 0x01 ) /* 그라데이션 존재여부 */ { padd( ascii("draw:name"), sXML_CDATA, ascii(Int2Str(hdo->index, "Grad%d", buf))); switch( prop->gstyle ) { case 1 : if( prop->center_y == 50 ) padd( ascii("draw:style"), sXML_CDATA, ascii("axial")); else padd( ascii("draw:style"), sXML_CDATA, ascii("linear")); break; case 2: case 3: padd( ascii("draw:style"), sXML_CDATA, ascii("radial")); break; case 4: padd( ascii("draw:style"), sXML_CDATA, ascii("square")); break; default: padd( ascii("draw:style"), sXML_CDATA, ascii("linear")); break; } padd( ascii("draw:cx"), sXML_CDATA,ascii(Int2Str(prop->center_x, "%d%%", buf))); padd( ascii("draw:cy"), sXML_CDATA,ascii(Int2Str(prop->center_y, "%d%%", buf))); HWPInfo& hwpinfo = hwpfile.GetHWPInfo(); int default_color = 0xffffff; if( hwpinfo.back_info.isset ) { if( hwpinfo.back_info.color[0] > 0 || hwpinfo.back_info.color[1] > 0 || hwpinfo.back_info.color[2] > 0 ) default_color = hwpinfo.back_info.color[0] << 16 | hwpinfo.back_info.color[1] << 8 | hwpinfo.back_info.color[2]; } if( prop->fromcolor > 0xffffff ) prop->fromcolor = default_color; if( prop->tocolor > 0xffffff ) prop->tocolor = default_color; if( prop->gstyle == 1) { if( prop->center_y == 100 ) { sprintf( buf, "#%02x%02x%02x", prop->tocolor & 0xff, (prop->tocolor >> 8) & 0xff, (prop->tocolor >>16) & 0xff ); padd( ascii("draw:start-color"), sXML_CDATA, ascii( buf )); sprintf( buf, "#%02x%02x%02x", prop->fromcolor & 0xff, (prop->fromcolor >> 8) & 0xff, (prop->fromcolor >>16) & 0xff ); padd( ascii("draw:end-color"), sXML_CDATA, ascii( buf )); } else { sprintf( buf, "#%02x%02x%02x", prop->fromcolor & 0xff, (prop->fromcolor >> 8) & 0xff, (prop->fromcolor >>16) & 0xff ); padd( ascii("draw:start-color"), sXML_CDATA, ascii( buf )); sprintf( buf, "#%02x%02x%02x", prop->tocolor & 0xff, (prop->tocolor >> 8) & 0xff, (prop->tocolor >>16) & 0xff ); padd( ascii("draw:end-color"), sXML_CDATA, ascii( buf )); } } else { sprintf( buf, "#%02x%02x%02x", prop->tocolor & 0xff, (prop->tocolor >> 8) & 0xff, (prop->tocolor >>16) & 0xff ); padd( ascii("draw:start-color"), sXML_CDATA,ascii( buf )); sprintf( buf, "#%02x%02x%02x", prop->fromcolor & 0xff, (prop->fromcolor >> 8) & 0xff, (prop->fromcolor >>16) & 0xff ); padd( ascii("draw:end-color"), sXML_CDATA,ascii( buf )); } if( prop->angle > 0 && ( prop->gstyle == 1 || prop->gstyle == 4)) { int angle = prop->angle >= 180 ? prop->angle - 180 : prop->angle; angle = 1800 - prop->angle * 10; padd( ascii("draw:angle"), sXML_CDATA, ascii(Int2Str( angle, "%d", buf))); } rstartEl( ascii("draw:gradient"), rList ); pList->clear(); rendEl( ascii("draw:gradient")); } /* 해칭 */ else if( prop->pattern_type >> 24 & 0x01 ) { int type = prop->pattern_type & 0xffffff; padd( ascii("draw:name"), sXML_CDATA, ascii(Int2Str(hdo->index, "Hatch%d", buf))); if( type < 4 ) padd( ascii("draw:style"), sXML_CDATA, ascii("single") ); else padd( ascii("draw:style"), sXML_CDATA, ascii("double") ); sprintf( buf, "#%02x%02x%02x", sal_uInt16(prop->pattern_color & 0xff), sal_uInt16((prop->pattern_color >> 8) & 0xff), sal_uInt16((prop->pattern_color >>16) & 0xff) ); padd( ascii("draw:color"), sXML_CDATA, ascii( buf )); padd( ascii("draw:distance"), sXML_CDATA, ascii("0.12cm")); switch( type ) { case 0 : case 4 : padd( ascii("draw:rotation"), sXML_CDATA, ascii("0")); break; case 1 : padd( ascii("draw:rotation"), sXML_CDATA, ascii("900")); break; case 2 : padd( ascii("draw:rotation"), sXML_CDATA, ascii("1350")); break; case 3 : case 5 : padd( ascii("draw:rotation"), sXML_CDATA, ascii("450")); break; } rstartEl( ascii("draw:hatch"), rList); pList->clear(); rendEl( ascii("draw:hatch")); } } hdo = hdo->next; } } void HwpReader::makeStyles() { HWPStyle& hwpstyle = hwpfile.GetHWPStyle(); rstartEl(ascii("office:styles"), rList); int i; for (i = 0; i < hwpfile.getFBoxStyleCount(); i++) { if( hwpfile.getFBoxStyle(i)->boxtype == 'D' ) { makeDrawMiscStyle((HWPDrawingObject *)hwpfile.getFBoxStyle(i)->cell ); } } padd(ascii("style:name"), sXML_CDATA, ascii("Standard")); padd(ascii("style:family"), sXML_CDATA, ascii("paragraph")); padd(ascii("style:class"), sXML_CDATA, ascii("text")); rstartEl(ascii("style:style"), rList); pList->clear(); padd(ascii("fo:line-height"), sXML_CDATA, ascii("160%")); padd(ascii("fo:text-align"), sXML_CDATA, ascii("justify")); rstartEl(ascii("style:properties"), rList); pList->clear(); rstartEl(ascii("style:tab-stops"), rList); for( i = 1 ; i < 40 ; i++) { padd(ascii("style:position"), sXML_CDATA, Double2Str( WTI(1000 * i)) + ascii("inch")); rstartEl(ascii("style:tab-stop"), rList); pList->clear(); rendEl(ascii("style:tab-stop")); } rendEl(ascii("style:tab-stops")); rendEl(ascii("style:properties")); rendEl(ascii("style:style")); for (int ii = 0; ii < hwpstyle.Num(); ii++) { unsigned char *stylename = (unsigned char *) hwpstyle.GetName(ii); padd(ascii("style:name"), sXML_CDATA, (hconv(kstr2hstr(stylename).c_str()))); padd(ascii("style:family"), sXML_CDATA, ascii("paragraph")); padd(ascii("style:parent-style-name"), sXML_CDATA, ascii("Standard")); rstartEl(ascii("style:style"), rList); pList->clear(); parseCharShape(hwpstyle.GetCharShape(ii)); parseParaShape(hwpstyle.GetParaShape(ii)); rstartEl(ascii("style:properties"), rList); pList->clear(); rendEl(ascii("style:properties")); rendEl(ascii("style:style")); } { padd( ascii("style:name"), sXML_CDATA, ascii("Header")); padd( ascii("style:family"), sXML_CDATA, ascii("paragraph")); padd( ascii("style:parent-style-name"), sXML_CDATA, ascii("Standard")); padd( ascii("style:class"), sXML_CDATA, ascii("extra")); rstartEl(ascii("style:style"), rList); pList->clear(); rendEl(ascii("style:style")); } { padd( ascii("style:name"), sXML_CDATA, ascii("Footer")); padd( ascii("style:family"), sXML_CDATA, ascii("paragraph")); padd( ascii("style:parent-style-name"), sXML_CDATA, ascii("Standard")); padd( ascii("style:class"), sXML_CDATA, ascii("extra")); rstartEl(ascii("style:style"), rList); pList->clear(); rendEl(ascii("style:style")); } if( hwpfile.linenumber > 0) { padd( ascii("style:name"), sXML_CDATA, ascii("Horizontal Line")); padd( ascii("style:family"), sXML_CDATA, ascii("paragraph")); padd( ascii("style:parent-style-name"), sXML_CDATA, ascii("Standard")); padd( ascii("style:class"), sXML_CDATA, ascii("html")); rstartEl( ascii("style:style"), rList); pList->clear(); padd( ascii("fo:font-size"), sXML_CDATA, ascii("6pt")); padd( ascii("fo:margin-top"), sXML_CDATA, ascii("0cm")); padd( ascii("fo:margin-bottom"), sXML_CDATA, ascii("0cm")); padd( ascii("style:border-line-width-bottom"), sXML_CDATA, ascii("0.02cm 0.035cm 0.002cm")); padd( ascii("fo:padding"), sXML_CDATA, ascii("0cm")); padd( ascii("fo:border-bottom"), sXML_CDATA, ascii("0.039cm double #808080")); padd( ascii("text:number-lines"), sXML_CDATA, ascii("false")); padd( ascii("text:line-number"), sXML_CDATA, ascii("0")); padd(ascii("fo:line-height"), sXML_CDATA, ascii("100%")); rstartEl( ascii("style:properties"), rList); pList->clear(); rendEl( ascii("style:properties")); rendEl( ascii("style:style")); } HWPInfo& hwpinfo = hwpfile.GetHWPInfo(); padd(ascii("text:num-suffix"), sXML_CDATA, ascii(")")); padd(ascii("text:num-format"), sXML_CDATA, ascii("1")); if( hwpinfo.beginfnnum != 1) padd(ascii("text:offset"), sXML_CDATA, ascii(Int2Str(hwpinfo.beginfnnum -1, "%d", buf))); rstartEl(ascii("text:footnotes-configuration"), rList); pList->clear(); rendEl(ascii("text:footnotes-configuration")); rendEl(ascii("office:styles")); } /** * parse automatic styles from hwpfile * 자동적으로 반영이 되는 스타일을 정의한다. 예를들어 각각의 문단이나, 테이블, 헤더 등등의 스타일을 이곳에서 정의하고, Body에서는 이곳에 정의된 스타일을 이용한다. * 1. paragraph, text, fbox, page스타일에 대해 지원한다. */ void HwpReader::makeAutoStyles() { int i; rstartEl(ascii("office:automatic-styles"), rList); for (i = 0; i < hwpfile.getParaShapeCount(); i++) makePStyle(hwpfile.getParaShape(i)); for (i = 0; i < hwpfile.getCharShapeCount(); i++) makeTStyle(hwpfile.getCharShape(i)); for( i = 0 ; i < hwpfile.getTableCount(); i++) makeTableStyle(hwpfile.getTable(i)); for (i = 0; i < hwpfile.getFBoxStyleCount(); i++) { if( hwpfile.getFBoxStyle(i)->boxtype == 'D' ) makeDrawStyle((HWPDrawingObject *)hwpfile.getFBoxStyle(i)->cell, hwpfile.getFBoxStyle(i)); else makeFStyle(hwpfile.getFBoxStyle(i)); } bool bIsLeft = false, bIsMiddle = false, bIsRight = false; for( i = 0 ; i < hwpfile.getPageNumberCount() ; i++ ) { ShowPageNum *pn = hwpfile.getPageNumber(i); if( pn->where == 7 || pn->where == 8 ) { bIsLeft = true; bIsRight = true; } else if( pn->where == 1 || pn->where == 4 ) { bIsLeft = true; } else if( pn->where == 2 || pn->where == 5 ) { bIsMiddle = true; } else if( pn->where == 3 || pn->where == 6 ) { bIsRight = true; } } for( i = 1; i <= 3 ; i++ ) { if( i == 1 && bIsLeft == false ) continue; if( i == 2 && bIsMiddle == false ) continue; if( i == 3 && bIsRight == false ) continue; padd(ascii("style:name"), sXML_CDATA, ascii(Int2Str(i,"PNPara%d", buf))); padd(ascii("style:family"), sXML_CDATA, ascii("paragraph")); padd(ascii("style:parent-style-name"), sXML_CDATA, ascii("Standard")); rstartEl(ascii("style:style"), rList); pList->clear(); if( i == 1 ) padd(ascii("fo:text-align"), sXML_CDATA, ascii("start")); else if ( i == 2 ) padd(ascii("fo:text-align"), sXML_CDATA, ascii("center")); else if ( i == 3 ) padd(ascii("fo:text-align"), sXML_CDATA, ascii("end")); rstartEl(ascii("style:properties"), rList); pList->clear(); rendEl( ascii("style:properties")); rendEl( ascii("style:style")); padd(ascii("style:name"), sXML_CDATA, ascii(Int2Str(i,"PNBox%d",buf))); padd(ascii("style:family"), sXML_CDATA, ascii("graphics")); rstartEl(ascii("style:style"), rList); pList->clear(); padd(ascii("fo:margin-top"), sXML_CDATA, ascii("0cm")); padd(ascii("fo:margin-bottom"), sXML_CDATA, ascii("0cm")); padd(ascii("style:wrap"), sXML_CDATA, ascii("run-through")); padd(ascii("style:vertical-pos"), sXML_CDATA, ascii("from-top")); padd(ascii("style:vertical-rel"), sXML_CDATA, ascii("paragraph")); if( i == 1 ) padd(ascii("style:horizontal-pos"), sXML_CDATA, ascii("left")); else if ( i == 2 ) padd(ascii("style:horizontal-pos"), sXML_CDATA, ascii("center")); else if ( i == 3 ) padd(ascii("style:horizontal-pos"), sXML_CDATA, ascii("right")); padd(ascii("style:horizontal-rel"), sXML_CDATA, ascii("paragraph")); padd(ascii("fo:padding"), sXML_CDATA, ascii("0cm")); padd(ascii("stylefamily"), sXML_CDATA, ascii("graphics")); rstartEl(ascii("style:properties"), rList); pList->clear(); rendEl(ascii("style:properties")); rendEl(ascii("style:style")); } for (i = 0; i < hwpfile.getDateFormatCount(); i++) makeDateFormat(hwpfile.getDateCode(i)); makePageStyle(); rendEl(ascii("office:automatic-styles")); } struct PageSetting { PageSetting() { header = 0L; header_odd = 0L; header_even = 0L; footer = 0L; footer_odd = 0L; footer_even = 0L; pagenumber=0L; bIsSet = false; } HeaderFooter *header ; HeaderFooter *header_odd ; HeaderFooter *header_even ; HeaderFooter *footer ; HeaderFooter *footer_odd ; HeaderFooter *footer_even ; ShowPageNum *pagenumber; bool bIsSet; }; void HwpReader::makeMasterStyles() { rstartEl(ascii("office:master-styles"), rList); int i; int nMax = hwpfile.getMaxSettedPage(); std::deque pSet(nMax + 1); for( i = 0 ; i < hwpfile.getPageNumberCount() ; i++ ) { ShowPageNum *pn = hwpfile.getPageNumber(i); pSet[pn->m_nPageNumber].pagenumber = pn; pSet[pn->m_nPageNumber].bIsSet = true; } for( i = 0 ; i < hwpfile.getHeaderFooterCount() ; i++ ) { HeaderFooter* hf = hwpfile.getHeaderFooter(i); pSet[hf->m_nPageNumber].bIsSet = true; if( hf->type == 0 ) // header { switch( hf->where ) { case 0 : pSet[hf->m_nPageNumber].header = hf; pSet[hf->m_nPageNumber].header_even = 0L; pSet[hf->m_nPageNumber].header_odd = 0L; break; case 1: pSet[hf->m_nPageNumber].header_even = hf; if( pSet[hf->m_nPageNumber].header ) { pSet[hf->m_nPageNumber].header_odd = pSet[hf->m_nPageNumber].header; pSet[hf->m_nPageNumber].header = 0L; } break; case 2: pSet[hf->m_nPageNumber].header_odd = hf; if( pSet[hf->m_nPageNumber].header ) { pSet[hf->m_nPageNumber].header_even = pSet[hf->m_nPageNumber].header; pSet[hf->m_nPageNumber].header = 0L; } break; } } else // footer { switch( hf->where ) { case 0 : pSet[hf->m_nPageNumber].footer = hf; pSet[hf->m_nPageNumber].footer_even = 0L; pSet[hf->m_nPageNumber].footer_odd = 0L; break; case 1: pSet[hf->m_nPageNumber].footer_even = hf; if( pSet[hf->m_nPageNumber].footer ) { pSet[hf->m_nPageNumber].footer_odd = pSet[hf->m_nPageNumber].footer; pSet[hf->m_nPageNumber].footer = 0L; } break; case 2: pSet[hf->m_nPageNumber].footer_odd = hf; if( pSet[hf->m_nPageNumber].footer ) { pSet[hf->m_nPageNumber].footer_even = pSet[hf->m_nPageNumber].footer; pSet[hf->m_nPageNumber].footer = 0L; } break; } } } PageSetting *pPrevSet = 0L; PageSetting *pPage = 0L; for( i = 1; i <= nMax ; i++ ) { if( i == 1 ) padd(ascii("style:name"), sXML_CDATA, ascii("Standard")); else padd(ascii("style:name"), sXML_CDATA, ascii(Int2Str(i, "p%d", buf))); padd(ascii("style:page-master-name"), sXML_CDATA, ascii(Int2Str(hwpfile.GetPageMasterNum(i), "pm%d", buf))); if( i < nMax ) padd(ascii("style:next-style-name"), sXML_CDATA, ascii(Int2Str(i+1, "p%d", buf))); padd(ascii("draw:style-name"), sXML_CDATA, ascii(Int2Str(i, "master%d", buf))); rstartEl(ascii("style:master-page"), rList); pList->clear(); if( pSet[i].bIsSet ) /* 현재 설정이 바뀌었으면 */ { if( !pSet[i].pagenumber ){ if( pPrevSet && pPrevSet->pagenumber ) pSet[i].pagenumber = pPrevSet->pagenumber; } if( pSet[i].pagenumber ) { if( pSet[i].pagenumber->where == 7 && pSet[i].header ) { pSet[i].header_even = pSet[i].header; pSet[i].header_odd = pSet[i].header; pSet[i].header = 0L; } if( pSet[i].pagenumber->where == 8 && pSet[i].footer ) { pSet[i].footer_even = pSet[i].footer; pSet[i].footer_odd = pSet[i].footer; pSet[i].footer = 0L; } } if( !pSet[i].header_even && pPrevSet && pPrevSet->header_even ) { pSet[i].header_even = pPrevSet->header_even; } if( !pSet[i].header_odd && pPrevSet && pPrevSet->header_odd ) { pSet[i].header_odd = pPrevSet->header_odd; } if( !pSet[i].footer_even && pPrevSet && pPrevSet->footer_even ) { pSet[i].footer_even = pPrevSet->footer_even; } if( !pSet[i].footer_odd && pPrevSet && pPrevSet->footer_odd ) { pSet[i].footer_odd = pPrevSet->footer_odd; } pPage = &pSet[i]; pPrevSet = &pSet[i]; } else if( pPrevSet ) /* 이전의 설정된 것이 있으면. */ { pPage = pPrevSet; } else /* 아직 설정이 없다면 기본설정으로 */ { rstartEl(ascii("style:header"), rList); padd(ascii("text:style-name"), sXML_CDATA, ascii("Standard")); rstartEl(ascii("text:p"), rList); pList->clear(); rendEl(ascii("text:p")); rendEl(ascii("style:header")); rstartEl(ascii("style:footer"), rList); padd(ascii("text:style-name"), sXML_CDATA, ascii("Standard")); rstartEl(ascii("text:p"), rList); pList->clear(); rendEl(ascii("text:p")); rendEl(ascii("style:footer")); rendEl(ascii("style:master-page")); continue; } // ------------- header ------------- if( pPage->header ) { rstartEl(ascii("style:header"), rList); if( pPage->pagenumber && pPage->pagenumber->where < 4 ) { d->bInHeader = true; d->pPn = pPage->pagenumber; } parsePara(pPage->header->plist.front()); d->bInHeader = false; d->pPn = 0L; rendEl(ascii("style:header")); } if( pPage->header_even ) { rstartEl(ascii("style:header"), rList); if( pPage->pagenumber && ( pPage->pagenumber->where < 4 || pPage->pagenumber->where == 7 ) ) { d->bInHeader = true; d->pPn = pPage->pagenumber; d->nPnPos = 3; } parsePara(pPage->header_even->plist.front()); d->bInHeader = false; d->pPn = 0L; d->nPnPos = 0; rendEl(ascii("style:header")); } /* 기본으로 한다. */ else if( pPage->header_odd && !pPage->header_even ) { rstartEl(ascii("style:header"), rList); padd(ascii("text:style-name"), sXML_CDATA, ascii("Standard")); rstartEl(ascii("text:p"), rList); pList->clear(); if( pPage->pagenumber && ( pPage->pagenumber->where < 4 || pPage->pagenumber->where == 7 ) ) { d->pPn = pPage->pagenumber; d->nPnPos = 3; makeShowPageNum(); d->pPn = 0L; d->nPnPos = 0; } rendEl(ascii("text:p")); rendEl(ascii("style:header")); } if( pPage->header_odd ) { rstartEl(ascii("style:header-left"), rList); if( pPage->pagenumber && ( pPage->pagenumber->where < 4 || pPage->pagenumber->where == 7 ) ) { d->bInHeader = true; d->nPnPos = 1; d->pPn = pPage->pagenumber; } parsePara(pPage->header_odd->plist.front()); d->bInHeader = false; d->pPn = 0L; d->nPnPos = 0; rendEl(ascii("style:header-left")); } /* 기본으로 한다. */ else if( pPage->header_even && !pPage->header_odd ) { rstartEl(ascii("style:header-left"), rList); padd(ascii("text:style-name"), sXML_CDATA, ascii("Standard")); rstartEl(ascii("text:p"), rList); pList->clear(); if( pPage->pagenumber && ( pPage->pagenumber->where < 4 || pPage->pagenumber->where == 7 ) ) { d->pPn = pPage->pagenumber; d->nPnPos = 1; makeShowPageNum(); d->pPn = 0L; d->nPnPos = 0; } rendEl(ascii("text:p")); rendEl(ascii("style:header-left")); } if( !pPage->header && !pPage->header_even && !pPage->header_odd ) { rstartEl(ascii("style:header"), rList); padd(ascii("text:style-name"), sXML_CDATA, ascii("Standard")); rstartEl(ascii("text:p"), rList); pList->clear(); if( pPage->pagenumber && (pPage->pagenumber->where < 4 || pPage->pagenumber->where == 7 ) ) { d->pPn = pPage->pagenumber; makeShowPageNum(); d->pPn = 0L; } rendEl(ascii("text:p")); rendEl(ascii("style:header")); } // ------------- footer ------------- if( pPage->footer ) { rstartEl(ascii("style:footer"), rList); if( pPage->pagenumber && pPage->pagenumber->where >= 4 && pPage->pagenumber->where != 7 ) { d->bInHeader = true; d->pPn = pPage->pagenumber; } parsePara(pPage->footer->plist.front()); d->bInHeader = false; d->pPn = 0L; rendEl(ascii("style:footer")); } if( pPage->footer_even ) { rstartEl(ascii("style:footer"), rList); if( pPage->pagenumber && pPage->pagenumber->where >= 4 && pPage->pagenumber->where != 7 ) { d->bInHeader = true; d->pPn = pPage->pagenumber; d->nPnPos = 3; } parsePara(pPage->footer_even->plist.front()); d->bInHeader = false; d->pPn = 0L; d->nPnPos = 0; rendEl(ascii("style:footer")); } /* 기본으로 한다. */ else if( pPage->footer_odd && !pPage->footer_even ) { rstartEl(ascii("style:footer"), rList); padd(ascii("text:style-name"), sXML_CDATA, ascii("Standard")); rstartEl(ascii("text:p"), rList); pList->clear(); if( pPage->pagenumber && pPage->pagenumber->where >= 4 && pPage->pagenumber->where != 7 ) { d->pPn = pPage->pagenumber; d->nPnPos = 3; makeShowPageNum(); d->pPn = 0L; d->nPnPos = 0; } rendEl(ascii("text:p")); rendEl(ascii("style:footer")); } if( pPage->footer_odd ) { rstartEl(ascii("style:footer-left"), rList); if( pPage->pagenumber && pPage->pagenumber->where >= 4 && pPage->pagenumber->where != 7 ) { d->bInHeader = true; d->pPn = pPage->pagenumber; d->nPnPos = 1; } parsePara(pPage->footer_odd->plist.front()); d->bInHeader = false; d->pPn = 0L; d->nPnPos = 0; rendEl(ascii("style:footer-left")); } /* 기본으로 한다. */ else if( pPage->footer_even && !pPage->footer_odd ) { rstartEl(ascii("style:footer-left"), rList); padd(ascii("text:style-name"), sXML_CDATA, ascii("Standard")); rstartEl(ascii("text:p"), rList); pList->clear(); if( pPage->pagenumber && pPage->pagenumber->where >= 4 && pPage->pagenumber->where != 7 ) { d->pPn = pPage->pagenumber; d->nPnPos = 1; makeShowPageNum(); d->pPn = 0L; d->nPnPos = 0; } rendEl(ascii("text:p")); rendEl(ascii("style:footer-left")); } if( !pPage->footer && !pPage->footer_even && !pPage->footer_odd ) { rstartEl(ascii("style:footer"), rList); padd(ascii("text:style-name"), sXML_CDATA, ascii("Standard")); rstartEl(ascii("text:p"), rList); pList->clear(); if( pPage->pagenumber && pPage->pagenumber->where >= 4 && pPage->pagenumber->where != 7 ) { d->pPn = pPage->pagenumber; makeShowPageNum(); d->pPn = 0L; } rendEl(ascii("text:p")); rendEl(ascii("style:footer")); } rendEl(ascii("style:master-page")); } rendEl(ascii("office:master-styles")); } /** * 텍스트 스타일을 위한 프로퍼티들을 만든다. * 1. fo:font-size, fo:font-family, fo:letter-spacing, fo:color, * style:text-background-color, fo:font-style, fo:font-weight, * style:text-underline,style:text-outline,fo:text-shadow,style:text-position * 을 지원한다. */ void HwpReader::parseCharShape(CharShape * cshape) { HWPFont& hwpfont = hwpfile.GetHWPFont(); padd(ascii("fo:font-size"), sXML_CDATA, ascii(Int2Str(cshape->size / 25, "%dpt", buf))); padd(ascii("style:font-size-asian"), sXML_CDATA, ascii(Int2Str(cshape->size / 25, "%dpt", buf))); ::std::string const tmp = hstr2ksstr(kstr2hstr( (unsigned char *) hwpfont.GetFontName(0, cshape->font[0])).c_str()); double fRatio = 1.0; int size = getRepFamilyName(tmp.c_str(), buf, fRatio); padd(ascii("fo:font-family"), sXML_CDATA, OUString(buf, size, RTL_TEXTENCODING_EUC_KR)); padd(ascii("style:font-family-asian"), sXML_CDATA, OUString(buf, size, RTL_TEXTENCODING_EUC_KR)); padd(ascii("style:text-scale"), sXML_CDATA, ascii(Int2Str((int)(cshape->ratio[0] * fRatio), "%d%%", buf))); double sspace = (cshape->size / 25) * cshape->space[0] / 100.; if (sspace != 0.) { padd(ascii("fo:letter-spacing"), sXML_CDATA, Double2Str(sspace) + ascii("pt")); } if (cshape->color[1] != 0) padd(ascii("fo:color"), sXML_CDATA, ascii(hcolor2str(cshape->color[1], 100, buf, true))); if (cshape->shade != 0) padd(ascii("style:text-background-color"), sXML_CDATA, ascii(hcolor2str(cshape->color[0], cshape->shade, buf))); if (cshape->attr & 0x01) { padd(ascii("fo:font-style"), sXML_CDATA, ascii("italic")); padd(ascii("style:font-style-asian"), sXML_CDATA, ascii("italic")); } else{ padd(ascii("fo:font-style"), sXML_CDATA, ascii("normal")); padd(ascii("style:font-style-asian"), sXML_CDATA, ascii("normal")); } if (cshape->attr >> 1 & 0x01) { padd(ascii("fo:font-weight"), sXML_CDATA, ascii("bold")); padd(ascii("style:font-weight-asian"), sXML_CDATA, ascii("bold")); } else{ padd(ascii("fo:font-weight"), sXML_CDATA, ascii("normal")); padd(ascii("style:font-weight-asian"), sXML_CDATA, ascii("normal")); } if (cshape->attr >> 2 & 0x01) padd(ascii("style:text-underline"), sXML_CDATA, ascii("single")); if (cshape->attr >> 3 & 0x01) padd(ascii("style:text-outline"), sXML_CDATA, ascii("true")); if (cshape->attr >> 4 & 0x01) padd(ascii("fo:text-shadow"), sXML_CDATA, ascii("1pt 1pt")); if (cshape->attr >> 5 & 0x01) padd(ascii("style:text-position"), sXML_CDATA, ascii("super 58%")); if (cshape->attr >> 6 & 0x01) padd(ascii("style:text-position"), sXML_CDATA, ascii("sub 58%")); } /** * 실제 Paragraph에 해당하는 properties들을 만든다. * 1. fo:margin-left,fo:margin-right,fo:margin-top, fo:margin-bottom, * fo:text-indent, fo:line-height, fo:text-align, fo:border * 가 구현됨. * TODO : 탭설정 => 기본값이 아닌것들만 선택적으로 설정해야 한다. */ void HwpReader::parseParaShape(ParaShape * pshape) { if (pshape->left_margin != 0) padd(ascii("fo:margin-left"), sXML_CDATA, Double2Str (WTI(pshape->left_margin )) + ascii("inch")); if (pshape->right_margin != 0) padd(ascii("fo:margin-right"), sXML_CDATA, Double2Str (WTI(pshape->right_margin)) + ascii("inch")); if (pshape->pspacing_prev != 0) padd(ascii("fo:margin-top"), sXML_CDATA, Double2Str (WTI(pshape->pspacing_prev)) + ascii("inch")); if (pshape->pspacing_next != 0) padd(ascii("fo:margin-bottom"), sXML_CDATA, Double2Str (WTI(pshape->pspacing_next)) + ascii("inch")); if (pshape->indent != 0) padd(ascii("fo:text-indent"), sXML_CDATA, Double2Str (WTI(pshape->indent)) + ascii("inch")); if (pshape->lspacing != 0) padd(ascii("fo:line-height"), sXML_CDATA, ascii(Int2Str (pshape->lspacing, "%d%%", buf))); unsigned char set_align = 0; switch ((int) pshape->arrange_type) { case 1: strcpy(buf, "start"); set_align = 1; break; case 2: strcpy(buf, "end"); set_align = 1; break; case 3: strcpy(buf, "center"); set_align = 1; break; case 4: case 5: case 6: strcpy(buf, "justify"); set_align = 1; break; } if (set_align) padd(ascii("fo:text-align"), sXML_CDATA, ascii(buf)); if (pshape->outline) padd(ascii("fo:border"), sXML_CDATA, ascii("0.002cm solid #000000")); if( pshape->shade > 0 ) { padd(ascii("fo:background-color"), sXML_CDATA, ascii(hcolor2str(0, pshape->shade, buf))); } if( pshape->pagebreak & 0x02 || pshape->pagebreak & 0x04) padd(ascii("fo:break-before"), sXML_CDATA, ascii("page")); else if( pshape->pagebreak & 0x01 ) padd(ascii("fo:break-before"), sXML_CDATA, ascii("column")); } /** * Paragraph에 대한 스타일을 만든다. */ void HwpReader::makePStyle(ParaShape * pshape) { int nscount = pshape->tabs[MAXTABS -1].type; padd(ascii("style:name"), sXML_CDATA, ascii(Int2Str(pshape->index, "P%d", buf))); padd(ascii("style:family"), sXML_CDATA, ascii("paragraph")); rstartEl(ascii("style:style"), rList); pList->clear(); parseParaShape(pshape); parseCharShape(pshape->cshape); rstartEl(ascii("style:properties"), rList); pList->clear(); if( nscount ) { unsigned char tf = 0; rstartEl(ascii("style:tab-stops"),rList); int tab_margin = pshape->left_margin + pshape->indent; if( tab_margin < 0 ) tab_margin = 0; for( int i = 0 ; i < MAXTABS -1 ; i++) { if( i > 0 && pshape->tabs[i].position == 0. ) break; if( pshape->tabs[i].position <= tab_margin ) continue; padd(ascii("style:position"), sXML_CDATA, Double2Str(WTMM(pshape->tabs[i].position - tab_margin )) + ascii("mm")); if( pshape->tabs[i].type ) { tf = 1; switch(pshape->tabs[i].type) { case 1 : padd(ascii("style:type"), sXML_CDATA, ascii("right")); break; case 2: padd(ascii("style:type"), sXML_CDATA, ascii("center")); break; case 3: padd(ascii("style:type"), sXML_CDATA, ascii("char")); padd(ascii("style:char"), sXML_CDATA, ascii(".")); break; } } if( pshape->tabs[i].dot_continue ) { tf = 1; padd(ascii("style:leader-char"), sXML_CDATA, ascii(".")); } rstartEl( ascii("style:tab-stop"), rList); pList->clear(); rendEl( ascii("style:tab-stop") ); if( (pshape->tabs[i].position != 1000 * i ) || tf ) { if( !--nscount ) break; } } rendEl( ascii("style:tab-stops")); } rendEl(ascii("style:properties")); rendEl(ascii("style:style")); } /** * 페이지의 스타일을 만든다. 여기에는 header/footer, footnote등이 포함된다. * TODO : , fo:background-color(정보가 없다) */ void HwpReader::makePageStyle() { HWPInfo& hwpinfo = hwpfile.GetHWPInfo(); int pmCount = hwpfile.getColumnCount(); for( int i = 0 ; i < pmCount ; i++ ){ padd(ascii("style:name"), sXML_CDATA, ascii(Int2Str(i + 1, "pm%d", buf))); rstartEl(ascii("style:page-master"),rList); pList->clear(); switch( hwpinfo.paper.paper_kind ) { case 3: // A4 if( hwpinfo.paper.paper_direction ) { padd(ascii("fo:page-height"),sXML_CDATA, ascii("210mm")); padd(ascii("fo:page-width"),sXML_CDATA, ascii("297mm")); } else { padd(ascii("fo:page-width"),sXML_CDATA, ascii("210mm")); padd(ascii("fo:page-height"),sXML_CDATA, ascii("297mm")); } break; case 4: // 80 column if( hwpinfo.paper.paper_direction ) { padd(ascii("fo:page-height"),sXML_CDATA, ascii("8.5inch")); padd(ascii("fo:page-width"),sXML_CDATA, ascii("11inch")); } else { padd(ascii("fo:page-width"),sXML_CDATA, ascii("8.5inch")); padd(ascii("fo:page-height"),sXML_CDATA, ascii("11inch")); } break; case 5: // B5 if( hwpinfo.paper.paper_direction ) { padd(ascii("fo:page-height"),sXML_CDATA, ascii("176mm")); padd(ascii("fo:page-width"),sXML_CDATA, ascii("250mm")); } else { padd(ascii("fo:page-width"),sXML_CDATA, ascii("176mm")); padd(ascii("fo:page-height"),sXML_CDATA, ascii("250mm")); } break; case 6: // B4 if( hwpinfo.paper.paper_direction ) { padd(ascii("fo:page-height"),sXML_CDATA, ascii("250mm")); padd(ascii("fo:page-width"),sXML_CDATA, ascii("353mm")); } else { padd(ascii("fo:page-width"),sXML_CDATA, ascii("250mm")); padd(ascii("fo:page-height"),sXML_CDATA, ascii("353mm")); } break; case 7: if( hwpinfo.paper.paper_direction ) { padd(ascii("fo:page-height"),sXML_CDATA, ascii("8.5inch")); padd(ascii("fo:page-width"),sXML_CDATA, ascii("14inch")); } else { padd(ascii("fo:page-width"),sXML_CDATA, ascii("8.5inch")); padd(ascii("fo:page-height"),sXML_CDATA, ascii("14inch")); } break; case 8: if( hwpinfo.paper.paper_direction ) { padd(ascii("fo:page-height"),sXML_CDATA, ascii("297mm")); padd(ascii("fo:page-width"),sXML_CDATA, ascii("420mm")); } else { padd(ascii("fo:page-width"),sXML_CDATA, ascii("297mm")); padd(ascii("fo:page-height"),sXML_CDATA, ascii("420mm")); } break; case 0: case 1: case 2: default: if( hwpinfo.paper.paper_direction ) { padd(ascii("fo:page-width"),sXML_CDATA, Double2Str(WTI(hwpinfo.paper.paper_height)) + ascii("inch")); padd(ascii("fo:page-height"),sXML_CDATA, Double2Str(WTI(hwpinfo.paper.paper_width)) + ascii("inch")); } else { padd(ascii("fo:page-width"),sXML_CDATA, Double2Str(WTI(hwpinfo.paper.paper_width)) + ascii("inch")); padd(ascii("fo:page-height"),sXML_CDATA, Double2Str(WTI(hwpinfo.paper.paper_height)) + ascii("inch")); } break; } padd(ascii("style:print-orientation"),sXML_CDATA, ascii(hwpinfo.paper.paper_direction ? "landscape" : "portrait")); if( hwpinfo.beginpagenum != 1) padd(ascii("style:first-page-number"),sXML_CDATA, ascii(Int2Str(hwpinfo.beginpagenum, "%d", buf))); if( hwpinfo.borderline ){ padd(ascii("fo:margin-left"),sXML_CDATA, Double2Str(WTI(hwpinfo.paper.left_margin - hwpinfo.bordermargin[0] + hwpinfo.paper.gutter_length)) + ascii("inch")); padd(ascii("fo:margin-right"),sXML_CDATA, Double2Str(WTI(hwpinfo.paper.right_margin - hwpinfo.bordermargin[1])) + ascii("inch")); padd(ascii("fo:margin-top"),sXML_CDATA, Double2Str(WTI(hwpinfo.paper.top_margin - hwpinfo.bordermargin[2])) + ascii("inch")); padd(ascii("fo:margin-bottom"),sXML_CDATA, Double2Str(WTI(hwpinfo.paper.bottom_margin - hwpinfo.bordermargin[3])) + ascii("inch")); } else{ padd(ascii("fo:margin-left"),sXML_CDATA, Double2Str(WTI(hwpinfo.paper.left_margin + hwpinfo.paper.gutter_length)) + ascii("inch")); padd(ascii("fo:margin-right"),sXML_CDATA, Double2Str(WTI(hwpinfo.paper.right_margin)) + ascii("inch")); padd(ascii("fo:margin-top"),sXML_CDATA, Double2Str(WTI(hwpinfo.paper.top_margin)) + ascii("inch")); padd(ascii("fo:margin-bottom"),sXML_CDATA, Double2Str(WTI(hwpinfo.paper.bottom_margin)) + ascii("inch")); } switch( hwpinfo.borderline ) { case 1: padd(ascii("fo:border"), sXML_CDATA,ascii("0.002cm solid #000000")); break; case 3: padd(ascii("fo:border"), sXML_CDATA,ascii("0.002cm dotted #000000")); break; case 2: padd(ascii("fo:border"), sXML_CDATA,ascii("0.035cm solid #000000")); break; case 4: padd(ascii("style:border-line-width"), sXML_CDATA,ascii("0.002cm 0.035cm 0.002cm")); padd(ascii("fo:border"), sXML_CDATA,ascii("0.039cm double #000000")); break; } padd(ascii("fo:padding-left"), sXML_CDATA, Double2Str(WTI(hwpinfo.bordermargin[0])) + ascii("inch")); padd(ascii("fo:padding-right"), sXML_CDATA, Double2Str(WTI(hwpinfo.bordermargin[1])) + ascii("inch")); padd(ascii("fo:padding-top"), sXML_CDATA, Double2Str(WTI(hwpinfo.bordermargin[2])) + ascii("inch")); padd(ascii("fo:padding-bottom"), sXML_CDATA, Double2Str(WTI(hwpinfo.bordermargin[3])) + ascii("inch")); /* background color */ if( hwpinfo.back_info.isset ) { if( hwpinfo.back_info.color[0] > 0 || hwpinfo.back_info.color[1] > 0 || hwpinfo.back_info.color[2] > 0 ){ sprintf(buf,"#%02x%02x%02x",hwpinfo.back_info.color[0], hwpinfo.back_info.color[1],hwpinfo.back_info.color[2] ); padd(ascii("fo:background-color"), sXML_CDATA, ascii(buf)); } } rstartEl(ascii("style:properties"),rList); pList->clear(); /* background image */ if( hwpinfo.back_info.isset && hwpinfo.back_info.type > 0 ) { if( hwpinfo.back_info.type == 1 ){ #ifdef _WIN32 padd(ascii("xlink:href"), sXML_CDATA, hconv(kstr2hstr((uchar*) urltowin(hwpinfo.back_info.filename).c_str()).c_str())); #else padd(ascii("xlink:href"), sXML_CDATA, hconv(kstr2hstr( (uchar *)urltounix(hwpinfo.back_info.filename).c_str()).c_str())); #endif padd(ascii("xlink:type"), sXML_CDATA, ascii("simple")); padd(ascii("xlink:actuate"), sXML_CDATA, ascii("onLoad")); } if( hwpinfo.back_info.flag >= 2) padd(ascii("style:repeat"), sXML_CDATA, ascii("stretch")); else if( hwpinfo.back_info.flag == 1 ){ padd(ascii("style:repeat"), sXML_CDATA, ascii("no-repeat")); padd(ascii("style:position"), sXML_CDATA, ascii("center")); } rstartEl(ascii("style:background-image"),rList); if( hwpinfo.back_info.type == 2 ){ rstartEl(ascii("office:binary-data"), rList); pList->clear(); boost::shared_ptr pStr(base64_encode_string((unsigned char *) hwpinfo.back_info.data, hwpinfo.back_info.size ), Free()); rchars(ascii(pStr.get())); rendEl(ascii("office:binary-data")); } rendEl(ascii("style:background-image")); } makeColumns( hwpfile.GetColumnDef(i) ); rendEl(ascii("style:properties")); /* header style */ rstartEl(ascii("style:header-style"), rList); padd(ascii("svg:height"), sXML_CDATA, Double2Str(WTI(hwpinfo.paper.header_length)) + ascii("inch")); padd(ascii("fo:margin-bottom"), sXML_CDATA, ascii("0mm")); rstartEl(ascii("style:properties"),rList); pList->clear(); rendEl(ascii("style:properties")); rendEl(ascii("style:header-style")); /* footer style */ rstartEl(ascii("style:footer-style"), rList); padd(ascii("svg:height"), sXML_CDATA, Double2Str(WTI(hwpinfo.paper.footer_length)) + ascii("inch")); padd(ascii("fo:margin-top"), sXML_CDATA, ascii("0mm")); rstartEl(ascii("style:properties"),rList); pList->clear(); rendEl(ascii("style:properties")); rendEl(ascii("style:footer-style")); /* footnote style 이건 dtd에서는 빠졌으나, 스펙에는 정의되어 있다. REALKING */ rstartEl(ascii("style:footnote-layout"), rList); padd(ascii("style:distance-before-sep"), sXML_CDATA, Double2Str(WTI(hwpinfo.splinetext)) + ascii("inch")); padd(ascii("style:distance-after-sep"), sXML_CDATA, Double2Str(WTI(hwpinfo.splinefn)) + ascii("inch")); rstartEl(ascii("style:properties"),rList); pList->clear(); rendEl(ascii("style:properties")); if ( hwpinfo.fnlinetype == 2 ) padd(ascii("style:width"), sXML_CDATA, ascii("15cm")); else if ( hwpinfo.fnlinetype == 1) padd(ascii("style:width"), sXML_CDATA, ascii("2cm")); else if ( hwpinfo.fnlinetype == 3) padd(ascii("style:width"), sXML_CDATA, ascii("0cm")); else padd(ascii("style:width"), sXML_CDATA, ascii("5cm")); rstartEl(ascii("style:footnote-sep"),rList); pList->clear(); rendEl(ascii("style:footnote-sep")); rendEl(ascii("style:footnote-layout")); rendEl(ascii("style:page-master")); } } void HwpReader::makeColumns(ColumnDef *coldef) { if( !coldef ) return; padd(ascii("fo:column-count"), sXML_CDATA, ascii(Int2Str(coldef->ncols, "%d", buf))); rstartEl(ascii("style:columns"),rList); pList->clear(); if( coldef->separator != 0 ) { switch( coldef->separator ) { case 1: /* 얇은선 */ padd(ascii("style:width"), sXML_CDATA, ascii("0.02mm")); //fall-through case 3: /* 점선 */ padd(ascii("style:style"), sXML_CDATA, ascii("dotted")); padd(ascii("style:width"), sXML_CDATA, ascii("0.02mm")); break; case 2: /* 두꺼운선 */ case 4: /* 2중선 */ padd(ascii("style:width"), sXML_CDATA, ascii("0.35mm")); break; case 0: /* 없음 */ default: padd(ascii("style:style"), sXML_CDATA, ascii("none")); break; } rstartEl(ascii("style:column-sep"),rList); pList->clear(); rendEl(ascii("style:column-sep")); } double spacing = WTI(coldef->spacing)/ 2. ; for(int ii = 0 ; ii < coldef->ncols ; ii++) { if( ii == 0 ) padd(ascii("fo:margin-left"), sXML_CDATA, ascii("0mm")); else padd(ascii("fo:margin-left"), sXML_CDATA, Double2Str( spacing) + ascii("inch")); if( ii == ( coldef->ncols -1) ) padd(ascii("fo:margin-right"), sXML_CDATA,ascii("0mm")); else padd(ascii("fo:margin-right"), sXML_CDATA, Double2Str( spacing) + ascii("inch")); rstartEl(ascii("style:column"),rList); pList->clear(); rendEl(ascii("style:column")); } rendEl(ascii("style:columns")); } void HwpReader::makeTStyle(CharShape * cshape) { padd(ascii("style:name"), sXML_CDATA, ascii(Int2Str(cshape->index, "T%d", buf))); padd(ascii("style:family"), sXML_CDATA, ascii("text")); rstartEl(ascii("style:style"), rList); pList->clear(); parseCharShape(cshape); rstartEl(ascii("style:properties"), rList); pList->clear(); rendEl(ascii("style:properties")); rendEl(ascii("style:style")); } void HwpReader::makeTableStyle(Table *tbl) { // --------------- table ---------------- TxtBox *hbox = tbl->box; padd(ascii("style:name"), sXML_CDATA, ascii(Int2Str(hbox->style.boxnum, "Table%d", buf))); padd(ascii("style:family"), sXML_CDATA,ascii("table")); rstartEl(ascii("style:style"), rList); pList->clear(); padd(ascii("style:width"), sXML_CDATA, Double2Str(WTMM(hbox->box_xs)) + ascii("mm")); padd(ascii("table:align"), sXML_CDATA,ascii("left")); padd(ascii("fo:keep-with-next"), sXML_CDATA,ascii("false")); rstartEl(ascii("style:properties"), rList); pList->clear(); rendEl(ascii("style:properties")); rendEl(ascii("style:style")); // --------------- column ---------------- for (size_t i = 0 ; i < tbl->columns.nCount -1 ; i++) { sprintf(buf,"Table%d.%c",hbox->style.boxnum, static_cast('A'+i)); padd(ascii("style:name"), sXML_CDATA, ascii( buf )); padd(ascii("style:family"), sXML_CDATA,ascii("table-column")); rstartEl(ascii("style:style"), rList); pList->clear(); padd(ascii("style:column-width"), sXML_CDATA, Double2Str(WTMM(tbl->columns.data[i+1] - tbl->columns.data[i])) + ascii("mm")); rstartEl(ascii("style:properties"), rList); pList->clear(); rendEl(ascii("style:properties")); rendEl(ascii("style:style")); } // --------------- row ---------------- for (size_t i = 0 ; i < tbl->rows.nCount -1 ; i++) { sprintf(buf,"Table%d.row%" SAL_PRI_SIZET "u",hbox->style.boxnum, i + 1); padd(ascii("style:name"), sXML_CDATA, ascii( buf )); padd(ascii("style:family"), sXML_CDATA,ascii("table-row")); rstartEl(ascii("style:style"), rList); pList->clear(); padd(ascii("style:row-height"), sXML_CDATA, Double2Str(WTMM(tbl->rows.data[i+1] - tbl->rows.data[i])) + ascii("mm")); rstartEl(ascii("style:properties"), rList); pList->clear(); rendEl(ascii("style:properties")); rendEl(ascii("style:style")); } // --------------- cell --------------------- for (std::list::iterator it = tbl->cells.begin(), aEnd = tbl->cells.end(); it != aEnd; ++it) { TCell *tcell = *it; sprintf(buf,"Table%d.%c%d",hbox->style.boxnum, 'A'+ tcell->nColumnIndex, tcell->nRowIndex +1); padd(ascii("style:name"), sXML_CDATA, ascii( buf )); padd(ascii("style:family"), sXML_CDATA,ascii("table-cell")); rstartEl(ascii("style:style"), rList); pList->clear(); Cell *cl = tcell->pCell; if( cl->ver_align == 1 ) padd(ascii("fo:vertical-align"), sXML_CDATA,ascii("middle")); if(cl->linetype[2] == cl->linetype[3] && cl->linetype[2] == cl->linetype[0] && cl->linetype[2] == cl->linetype[1]) { switch( cl->linetype[2] ) { case 1: /* 가는실선 */ case 3: /* 점선 -> 스타오피스에는 점선이 없다. */ padd(ascii("fo:border"), sXML_CDATA,ascii("0.002cm solid #000000")); break; case 2: /* 굵은실선 */ padd(ascii("fo:border"), sXML_CDATA,ascii("0.035cm solid #000000")); break; case 4: /* 2중선 */ padd(ascii("style:border-line-width"), sXML_CDATA,ascii("0.002cm 0.035cm 0.002cm")); padd(ascii("fo:border"), sXML_CDATA,ascii("0.039cm double #000000")); break; } } else { switch( cl->linetype[0] ) { case 1: /* 가는실선 */ case 3: /* 점선 -> 스타오피스에는 점선이 없다. */ padd(ascii("fo:border-left"), sXML_CDATA,ascii("0.002cm solid #000000")); break; case 2: /* 굵은실선 */ padd(ascii("fo:border-left"), sXML_CDATA,ascii("0.035cm solid #000000")); break; case 4: /* 2중선 */ padd(ascii("style:border-line-width-left"), sXML_CDATA,ascii("0.002cm 0.035cm 0.002cm")); padd(ascii("fo:border-left"), sXML_CDATA,ascii("0.039cm double #000000")); break; } switch( cl->linetype[1] ) { case 1: /* 가는실선 */ case 3: /* 점선 -> 스타오피스에는 점선이 없다. */ padd(ascii("fo:border-right"), sXML_CDATA,ascii("0.002cm solid #000000")); break; case 2: /* 굵은실선 */ padd(ascii("fo:border-right"), sXML_CDATA,ascii("0.035cm solid #000000")); break; case 4: /* 2중선 */ padd(ascii("style:border-line-width-right"), sXML_CDATA,ascii("0.002cm 0.035cm 0.002cm")); padd(ascii("fo:border-right"), sXML_CDATA,ascii("0.039cm double #000000")); break; } switch( cl->linetype[2] ) { case 1: /* 가는실선 */ case 3: /* 점선 -> 스타오피스에는 점선이 없다. */ padd(ascii("fo:border-top"), sXML_CDATA,ascii("0.002cm solid #000000")); break; case 2: /* 굵은실선 */ padd(ascii("fo:border-top"), sXML_CDATA,ascii("0.035cm solid #000000")); break; case 4: /* 2중선 */ padd(ascii("style:border-line-width-top"), sXML_CDATA,ascii("0.002cm 0.035cm 0.002cm")); padd(ascii("fo:border-top"), sXML_CDATA,ascii("0.039cm double #000000")); break; } switch( cl->linetype[3] ) { case 1: /* 가는실선 */ case 3: /* 점선 -> 스타오피스에는 점선이 없다. */ padd(ascii("fo:border-bottom"), sXML_CDATA,ascii("0.002cm solid #000000")); break; case 2: /* 굵은실선 */ padd(ascii("fo:border-bottom"), sXML_CDATA,ascii("0.035cm solid #000000")); break; case 4: /* 2중선 */ padd(ascii("style:border-line-width-bottom"), sXML_CDATA,ascii("0.002cm 0.035cm 0.002cm")); padd(ascii("fo:border-bottom"), sXML_CDATA,ascii("0.039cm double #000000")); break; } } if(cl->shade != 0) padd(ascii("fo:background-color"), sXML_CDATA, ascii(hcolor2str(sal::static_int_cast(cl->color), sal::static_int_cast(cl->shade), buf))); rstartEl(ascii("style:properties"), rList); pList->clear(); rendEl(ascii("style:properties")); rendEl(ascii("style:style")); } } void HwpReader::makeDrawStyle( HWPDrawingObject * hdo, FBoxStyle * fstyle) { while( hdo ) { padd(ascii("style:name"), sXML_CDATA, ascii(Int2Str(hdo->index, "Draw%d", buf))); padd(ascii("style:family"), sXML_CDATA, ascii("graphics")); rstartEl(ascii("style:style"), rList); pList->clear(); switch (fstyle->txtflow) { case 0: break; case 1: padd(ascii("style:wrap"), sXML_CDATA, ascii("run-through")); break; case 2: padd(ascii("style:wrap"), sXML_CDATA, ascii("dynamic")); break; } long color; // invisible line if( hdo->property.line_color > 0xffffff ) { padd(ascii("draw:stroke"), sXML_CDATA, ascii("none") ); } else { if( hdo->property.line_pstyle == 0 ) padd(ascii("draw:stroke"), sXML_CDATA, ascii("solid") ); else if( hdo->property.line_pstyle < 5 ) { padd(ascii("draw:stroke"), sXML_CDATA, ascii("dash") ); padd(ascii("draw:stroke-dash"), sXML_CDATA, ascii(Int2Str(hdo->index, "LineType%d", buf))); } padd(ascii("svg:stroke-width"), sXML_CDATA, Double2Str( WTMM(hdo->property.line_width)) + ascii("mm" )); color = hdo->property.line_color; sprintf( buf, "#%02x%02x%02x", sal_uInt16(color & 0xff), sal_uInt16((color >> 8) & 0xff), sal_uInt16((color >>16) & 0xff) ); padd(ascii("svg:stroke-color"), sXML_CDATA, ascii( buf) ); } if( hdo->type == HWPDO_LINE || hdo->type == HWPDO_ARC || hdo->type == HWPDO_FREEFORM || hdo->type == HWPDO_ADVANCED_ARC ) { if( hdo->property.line_tstyle > 0 ) { padd(ascii("draw:marker-start"), sXML_CDATA, ascii(ArrowShape[hdo->property.line_tstyle].name) ); if( hdo->property.line_width > 100 ) padd(ascii("draw:marker-start-width"), sXML_CDATA, Double2Str( WTMM(hdo->property.line_width * 3)) + ascii("mm" )); else if( hdo->property.line_width > 80 ) padd(ascii("draw:marker-start-width"), sXML_CDATA, Double2Str( WTMM(hdo->property.line_width * 4)) + ascii("mm" )); else if( hdo->property.line_width > 60 ) padd(ascii("draw:marker-start-width"), sXML_CDATA, Double2Str( WTMM(hdo->property.line_width * 5)) + ascii("mm" )); else if( hdo->property.line_width > 40 ) padd(ascii("draw:marker-start-width"), sXML_CDATA, Double2Str( WTMM(hdo->property.line_width * 6)) + ascii("mm" )); else padd(ascii("draw:marker-start-width"), sXML_CDATA, Double2Str( WTMM(hdo->property.line_width * 7)) + ascii("mm" )); } if( hdo->property.line_hstyle > 0 ) { padd(ascii("draw:marker-end"), sXML_CDATA, ascii(ArrowShape[hdo->property.line_hstyle].name) ); if( hdo->property.line_width > 100 ) padd(ascii("draw:marker-end-width"), sXML_CDATA, Double2Str( WTMM(hdo->property.line_width * 3)) + ascii("mm" )); else if( hdo->property.line_width > 80 ) padd(ascii("draw:marker-end-width"), sXML_CDATA, Double2Str( WTMM(hdo->property.line_width * 4)) + ascii("mm" )); else if( hdo->property.line_width > 60 ) padd(ascii("draw:marker-end-width"), sXML_CDATA, Double2Str( WTMM(hdo->property.line_width * 5)) + ascii("mm" )); else if( hdo->property.line_width > 40 ) padd(ascii("draw:marker-end-width"), sXML_CDATA, Double2Str( WTMM(hdo->property.line_width * 6)) + ascii("mm" )); else padd(ascii("draw:marker-end-width"), sXML_CDATA, Double2Str( WTMM(hdo->property.line_width * 7)) + ascii("mm" )); } } if(hdo->type != HWPDO_LINE ) { if( hdo->property.flag >> 19 & 0x01 ) { padd( ascii("draw:textarea-horizontal-align"), sXML_CDATA, ascii("center")); } color = hdo->property.fill_color; if( hdo->property.flag >> 18 & 0x01 ) // bitmap pattern { padd(ascii("draw:fill"), sXML_CDATA, ascii("bitmap")); padd(ascii("draw:fill-image-name"), sXML_CDATA, ascii(Int2Str(hdo->index, "fillimage%d", buf))); // bitmap resizing if( hdo->property.flag >> 3 & 0x01 ) { padd(ascii("style:repeat"), sXML_CDATA, ascii("stretch")); } else { padd(ascii("style:repeat"), sXML_CDATA, ascii("repeat")); padd(ascii("draw:fill-image-ref-point"), sXML_CDATA, ascii("top-left")); } if( hdo->property.flag >> 20 & 0x01 ) { if( hdo->property.luminance > 0 ) { padd(ascii("draw:transparency"), sXML_CDATA, ascii(Int2Str(hdo->property.luminance, "%d%%", buf))); } } } // Gradation else if( hdo->property.flag >> 16 & 0x01 ) { padd(ascii("draw:fill"), sXML_CDATA, ascii("gradient")); padd(ascii("draw:fill-gradient-name"), sXML_CDATA, ascii(Int2Str(hdo->index, "Grad%d", buf))); padd(ascii("draw:gradient-step-count"), sXML_CDATA, ascii(Int2Str(hdo->property.nstep, "%d", buf))); } // Hatching else if( hdo->property.pattern_type >> 24 & 0x01 ) { padd(ascii("draw:fill"), sXML_CDATA, ascii("hatch")); padd(ascii("draw:fill-hatch-name"), sXML_CDATA, ascii(Int2Str(hdo->index, "Hatch%d", buf))); if( color < 0xffffff ) { sprintf( buf, "#%02x%02x%02x", sal_uInt16(color & 0xff), sal_uInt16((color >> 8) & 0xff), sal_uInt16((color >>16) & 0xff) ); padd(ascii("draw:fill-color"), sXML_CDATA, ascii( buf) ); padd(ascii("draw:fill-hatch-solid"), sXML_CDATA, ascii("true")); } } else if( color <= 0xffffff ) { padd(ascii("draw:fill"), sXML_CDATA, ascii("solid")); sprintf( buf, "#%02x%02x%02x", sal_uInt16(color & 0xff), sal_uInt16((color >> 8) & 0xff), sal_uInt16((color >>16) & 0xff) ); padd(ascii("draw:fill-color"), sXML_CDATA, ascii( buf) ); } else padd(ascii("draw:fill"), sXML_CDATA, ascii("none")); } if( fstyle->anchor_type == CHAR_ANCHOR ) { padd(ascii("style:vertical-pos"), sXML_CDATA, ascii("top")); padd(ascii("style:vertical-rel"), sXML_CDATA, ascii("baseline")); } rstartEl(ascii("style:properties"), rList); pList->clear(); rendEl(ascii("style:properties")); rendEl(ascii("style:style")); if( hdo->type == 0 ) { makeDrawStyle( hdo->child, fstyle ); } hdo = hdo->next; } } void HwpReader::makeCaptionStyle(FBoxStyle * fstyle) { padd(ascii("style:name"), sXML_CDATA, ascii(Int2Str(fstyle->boxnum, "CapBox%d", buf))); padd(ascii("style:family"), sXML_CDATA, ascii("graphics")); rstartEl(ascii("style:style"), rList); pList->clear(); padd(ascii("fo:margin-left"), sXML_CDATA, ascii("0cm")); padd(ascii("fo:margin-right"), sXML_CDATA, ascii("0cm")); padd(ascii("fo:margin-top"), sXML_CDATA, ascii("0cm")); padd(ascii("fo:margin-bottom"), sXML_CDATA, ascii("0cm")); padd(ascii("fo:padding"), sXML_CDATA, ascii("0cm")); switch (fstyle->txtflow) { case 0: padd(ascii("style:wrap"), sXML_CDATA, ascii("none")); break; case 1: if( fstyle->boxtype == 'G' ) padd(ascii("style:run-through"), sXML_CDATA, ascii("background")); padd(ascii("style:wrap"), sXML_CDATA, ascii("run-through")); break; case 2: padd(ascii("style:wrap"), sXML_CDATA, ascii("dynamic")); break; } if (fstyle->anchor_type == CHAR_ANCHOR) { padd(ascii("style:vertical-pos"), sXML_CDATA, ascii("top")); padd(ascii("style:vertical-rel"), sXML_CDATA, ascii("baseline")); padd(ascii("style:horizontal-pos"), sXML_CDATA, ascii("center")); padd(ascii("style:horizontal-rel"), sXML_CDATA, ascii("paragraph")); } else { switch (-(fstyle->xpos)) { case 2: padd(ascii("style:horizontal-pos"), sXML_CDATA, ascii("right")); break; case 3: padd(ascii("style:horizontal-pos"), sXML_CDATA, ascii("center")); break; case 1: default: padd(ascii("style:horizontal-pos"), sXML_CDATA, ascii("from-left")); break; } switch (-(fstyle->ypos)) { case 2: padd(ascii("style:vertical-pos"), sXML_CDATA, ascii("bottom")); break; case 3: padd(ascii("style:vertical-pos"), sXML_CDATA, ascii("middle")); break; case 1: default: padd(ascii("style:vertical-pos"), sXML_CDATA, ascii("from-top")); break; } if ( fstyle->anchor_type == PARA_ANCHOR ) { padd(ascii("style:vertical-rel"), sXML_CDATA, ascii("paragraph")); padd(ascii("style:horizontal-rel"), sXML_CDATA, ascii("paragraph")); } else { padd(ascii("style:vertical-rel"), sXML_CDATA, ascii("page-content")); padd(ascii("style:horizontal-rel"), sXML_CDATA, ascii("page-content")); } } rstartEl(ascii("style:properties"), rList); pList->clear(); rendEl(ascii("style:properties")); rendEl(ascii("style:style")); if( fstyle->boxtype == 'G' ) { padd(ascii("style:name"), sXML_CDATA, ascii(Int2Str(fstyle->boxnum, "G%d", buf))); } else { padd(ascii("style:name"), sXML_CDATA, ascii(Int2Str(fstyle->boxnum, "Txtbox%d", buf))); } padd(ascii("style:family"), sXML_CDATA, ascii("graphics")); rstartEl(ascii("style:style"), rList); pList->clear(); padd(ascii("fo:margin-left"), sXML_CDATA, ascii("0cm")); padd(ascii("fo:margin-right"), sXML_CDATA, ascii("0cm")); padd(ascii("fo:margin-top"), sXML_CDATA, ascii("0cm")); padd(ascii("fo:margin-bottom"), sXML_CDATA, ascii("0cm")); padd(ascii("fo:padding"), sXML_CDATA, ascii("0cm")); padd(ascii("style:wrap"), sXML_CDATA, ascii("none")); padd(ascii("style:vertical-pos"), sXML_CDATA, ascii("from-top")); padd(ascii("style:vertical-rel"), sXML_CDATA, ascii("paragraph")); padd(ascii("style:horizontal-pos"), sXML_CDATA, ascii("from-left")); padd(ascii("style:horizontal-rel"), sXML_CDATA, ascii("paragraph")); if( fstyle->boxtype == 'G' ) { char *cell = (char *)fstyle->cell; padd(ascii("draw:luminance"), sXML_CDATA, ascii(Int2Str(cell[0], "%d%%", buf))); padd(ascii("draw:contrast"), sXML_CDATA, ascii(Int2Str(cell[1], "%d%%", buf))); if( cell[2] == 0 ) padd(ascii("draw:color-mode"), sXML_CDATA, ascii("standard")); else if( cell[2] == 1 ) padd(ascii("draw:color-mode"), sXML_CDATA, ascii("greyscale")); else if( cell[2] == 2 ) padd(ascii("draw:color-mode"), sXML_CDATA, ascii("mono")); } else { Cell *cell = (Cell *)fstyle->cell; if(cell->linetype[0] == cell->linetype[1] && cell->linetype[0] == cell->linetype[2] && cell->linetype[0] == cell->linetype[3]) { switch( cell->linetype[0] ) { case 0: padd(ascii("fo:padding"), sXML_CDATA,ascii("0mm")); break; case 1: /* 가는실선 */ case 3: /* 점선 -> 스타오피스에는 점선이 없다. */ padd(ascii("fo:border"), sXML_CDATA,ascii("0.002cm solid #000000")); break; case 2: /* 굵은실선 */ padd(ascii("fo:border"), sXML_CDATA,ascii("0.035cm solid #000000")); break; case 4: /* 2중선 */ padd(ascii("style:border-line-width"), sXML_CDATA,ascii("0.002cm 0.035cm 0.002cm")); padd(ascii("fo:border"), sXML_CDATA,ascii("0.039cm double #000000")); break; } } else { switch( cell->linetype[0] ) { case 1: /* 가는실선 */ case 3: /* 점선 -> 스타오피스에는 점선이 없다. */ padd(ascii("fo:border-left"), sXML_CDATA,ascii("0.002cm solid #000000")); break; case 2: /* 굵은실선 */ padd(ascii("fo:border-left"), sXML_CDATA,ascii("0.035cm solid #000000")); break; case 4: /* 2중선 */ padd(ascii("style:border-line-width-left"), sXML_CDATA,ascii("0.002cm 0.035cm 0.002cm")); padd(ascii("fo:border-left"), sXML_CDATA,ascii("0.039cm double #000000")); break; } switch( cell->linetype[1] ) { case 1: /* 가는실선 */ case 3: /* 점선 -> 스타오피스에는 점선이 없다. */ padd(ascii("fo:border-right"), sXML_CDATA,ascii("0.002cm solid #000000")); break; case 2: /* 굵은실선 */ padd(ascii("fo:border-right"), sXML_CDATA,ascii("0.035cm solid #000000")); break; case 4: /* 2중선 */ padd(ascii("style:border-line-width-right"), sXML_CDATA,ascii("0.002cm 0.035cm 0.002cm")); padd(ascii("fo:border-right"), sXML_CDATA,ascii("0.039cm double #000000")); break; } switch( cell->linetype[2] ) { case 1: /* 가는실선 */ case 3: /* 점선 -> 스타오피스에는 점선이 없다. */ padd(ascii("fo:border-top"), sXML_CDATA,ascii("0.002cm solid #000000")); break; case 2: /* 굵은실선 */ padd(ascii("fo:border-top"), sXML_CDATA,ascii("0.035cm solid #000000")); break; case 4: /* 2중선 */ padd(ascii("style:border-line-width-top"), sXML_CDATA,ascii("0.002cm 0.035cm 0.002cm")); padd(ascii("fo:border-top"), sXML_CDATA,ascii("0.039cm double #000000")); break; } switch( cell->linetype[3] ) { case 1: /* 가는실선 */ case 3: /* 점선 -> 스타오피스에는 점선이 없다. */ padd(ascii("fo:border-bottom"), sXML_CDATA,ascii("0.002cm solid #000000")); break; case 2: /* 굵은실선 */ padd(ascii("fo:border-bottom"), sXML_CDATA,ascii("0.035cm solid #000000")); break; case 4: /* 2중선 */ padd(ascii("style:border-line-width-bottom"), sXML_CDATA,ascii("0.002cm 0.035cm 0.002cm")); padd(ascii("fo:border-bottom"), sXML_CDATA,ascii("0.039cm double #000000")); break; } } if(cell->shade != 0) padd(ascii("fo:background-color"), sXML_CDATA, ascii(hcolor2str( sal::static_int_cast(cell->color), sal::static_int_cast(cell->shade), buf))); } rstartEl(ascii("style:properties"), rList); pList->clear(); rendEl(ascii("style:properties")); rendEl(ascii("style:style")); } /** * Floating 객체에 대한 스타일을 만든다. */ void HwpReader::makeFStyle(FBoxStyle * fstyle) { /* 캡션 exist */ if( ( fstyle->boxtype == 'G' || fstyle->boxtype == 'X' ) && fstyle->cap_len > 0 ) { makeCaptionStyle(fstyle); return; } switch( fstyle->boxtype ) { case 'X' : // txtbox case 'E' : // equation case 'B' : // button case 'O' : // other case 'T' : // table padd(ascii("style:name"), sXML_CDATA, ascii(Int2Str(fstyle->boxnum, "Txtbox%d", buf))); padd(ascii("style:family"), sXML_CDATA, ascii("graphics")); break; case 'G' : // graphics padd(ascii("style:name"), sXML_CDATA, ascii(Int2Str(fstyle->boxnum, "G%d", buf))); padd(ascii("style:family"), sXML_CDATA, ascii("graphics")); break; case 'L' : // line TODO : all padd(ascii("style:name"), sXML_CDATA, ascii(Int2Str(fstyle->boxnum, "L%d", buf))); padd( ascii("style:family") , sXML_CDATA , ascii("paragraph") ); break; } rstartEl(ascii("style:style"), rList); pList->clear(); if ( fstyle->boxtype == 'T') { padd(ascii("fo:padding"), sXML_CDATA, ascii("0cm")); } if( !(fstyle->boxtype == 'G' && fstyle->cap_len > 0 )) { padd(ascii("fo:margin-left"), sXML_CDATA, Double2Str(WTMM(fstyle->margin[0][0]) ) + ascii("mm")); padd(ascii("fo:margin-right"), sXML_CDATA, Double2Str(WTMM(fstyle->margin[0][1])) + ascii("mm")); padd(ascii("fo:margin-top"), sXML_CDATA, Double2Str(WTMM(fstyle->margin[0][2])) + ascii("mm")); padd(ascii("fo:margin-bottom"), sXML_CDATA, Double2Str(WTMM(fstyle->margin[0][3])) + ascii("mm")); } switch (fstyle->txtflow) { case 0: padd(ascii("style:wrap"), sXML_CDATA, ascii("none")); break; case 1: if( fstyle->boxtype == 'G' || fstyle->boxtype == 'B' || fstyle->boxtype == 'O') padd(ascii("style:run-through"), sXML_CDATA, ascii("background")); padd(ascii("style:wrap"), sXML_CDATA, ascii("run-through")); break; case 2: padd(ascii("style:wrap"), sXML_CDATA, ascii("dynamic")); break; } if (fstyle->anchor_type == CHAR_ANCHOR) { padd(ascii("style:vertical-pos"), sXML_CDATA, ascii("top")); padd(ascii("style:vertical-rel"), sXML_CDATA, ascii("baseline")); padd(ascii("style:horizontal-pos"), sXML_CDATA, ascii("center")); padd(ascii("style:horizontal-rel"), sXML_CDATA, ascii("paragraph")); } else { switch (-(fstyle->xpos)) { case 2: padd(ascii("style:horizontal-pos"), sXML_CDATA, ascii("right")); break; case 3: padd(ascii("style:horizontal-pos"), sXML_CDATA, ascii("center")); break; case 1: default: padd(ascii("style:horizontal-pos"), sXML_CDATA, ascii("from-left")); break; } switch (-(fstyle->ypos)) { case 2: padd(ascii("style:vertical-pos"), sXML_CDATA, ascii("bottom")); break; case 3: padd(ascii("style:vertical-pos"), sXML_CDATA, ascii("middle")); break; case 1: default: padd(ascii("style:vertical-pos"), sXML_CDATA, ascii("from-top")); break; } if ( fstyle->anchor_type == PARA_ANCHOR ) { padd(ascii("style:vertical-rel"), sXML_CDATA, ascii("paragraph")); padd(ascii("style:horizontal-rel"), sXML_CDATA, ascii("paragraph")); } else { padd(ascii("style:vertical-rel"), sXML_CDATA, ascii("page-content")); padd(ascii("style:horizontal-rel"), sXML_CDATA, ascii("page-content")); } } if( fstyle->boxtype == 'X' || fstyle->boxtype == 'B' ) { Cell *cell = (Cell *)fstyle->cell; if(cell->linetype[0] == cell->linetype[1] && cell->linetype[0] == cell->linetype[2] && cell->linetype[0] == cell->linetype[3]) { switch( cell->linetype[0] ) { case 0: padd(ascii("fo:border"), sXML_CDATA, ascii("none")); break; case 1: /* 가는실선 */ case 3: /* 점선 -> 스타오피스에는 점선이 없다. */ padd(ascii("fo:border"), sXML_CDATA,ascii("0.002cm solid #000000")); break; case 2: /* 굵은실선 */ padd(ascii("fo:border"), sXML_CDATA,ascii("0.035cm solid #000000")); break; case 4: /* 2중선 */ padd(ascii("style:border-line-width"), sXML_CDATA,ascii("0.002cm 0.035cm 0.002cm")); padd(ascii("fo:border"), sXML_CDATA,ascii("0.039cm double #000000")); break; } } else { switch( cell->linetype[0] ) { case 1: /* 가는실선 */ case 3: /* 점선 -> 스타오피스에는 점선이 없다. */ padd(ascii("fo:border-left"), sXML_CDATA,ascii("0.002cm solid #000000")); break; case 2: /* 굵은실선 */ padd(ascii("fo:border-left"), sXML_CDATA,ascii("0.035cm solid #000000")); break; case 4: /* 2중선 */ padd(ascii("style:border-line-width-left"), sXML_CDATA,ascii("0.002cm 0.035cm 0.002cm")); padd(ascii("fo:border-left"), sXML_CDATA,ascii("0.039cm double #000000")); break; } switch( cell->linetype[1] ) { case 1: /* 가는실선 */ case 3: /* 점선 -> 스타오피스에는 점선이 없다. */ padd(ascii("fo:border-right"), sXML_CDATA,ascii("0.002cm solid #000000")); break; case 2: /* 굵은실선 */ padd(ascii("fo:border-right"), sXML_CDATA,ascii("0.035cm solid #000000")); break; case 4: /* 2중선 */ padd(ascii("style:border-line-width-right"), sXML_CDATA,ascii("0.002cm 0.035cm 0.002cm")); padd(ascii("fo:border-right"), sXML_CDATA,ascii("0.039cm double #000000")); break; } switch( cell->linetype[2] ) { case 1: /* 가는실선 */ case 3: /* 점선 -> 스타오피스에는 점선이 없다. */ padd(ascii("fo:border-top"), sXML_CDATA,ascii("0.002cm solid #000000")); break; case 2: /* 굵은실선 */ padd(ascii("fo:border-top"), sXML_CDATA,ascii("0.035cm solid #000000")); break; case 4: /* 2중선 */ padd(ascii("style:border-line-width-top"), sXML_CDATA,ascii("0.002cm 0.035cm 0.002cm")); padd(ascii("fo:border-top"), sXML_CDATA,ascii("0.039cm double #000000")); break; } switch( cell->linetype[3] ) { case 1: /* 가는실선 */ case 3: /* 점선 -> 스타오피스에는 점선이 없다. */ padd(ascii("fo:border-bottom"), sXML_CDATA,ascii("0.002cm solid #000000")); break; case 2: /* 굵은실선 */ padd(ascii("fo:border-bottom"), sXML_CDATA,ascii("0.035cm solid #000000")); break; case 4: /* 2중선 */ padd(ascii("style:border-line-width-bottom"), sXML_CDATA,ascii("0.002cm 0.035cm 0.002cm")); padd(ascii("fo:border-bottom"), sXML_CDATA,ascii("0.039cm double #000000")); break; } } if( cell->linetype[0] == 0 && cell->linetype[1] == 0 && cell->linetype[2] == 0 && cell->linetype[3] == 0 ){ padd(ascii("fo:padding"), sXML_CDATA,ascii("0mm")); } else{ padd(ascii("fo:padding-left"), sXML_CDATA, Double2Str(WTMM(fstyle->margin[1][0])) + ascii("mm")); padd(ascii("fo:padding-right"), sXML_CDATA, Double2Str(WTMM(fstyle->margin[1][1])) + ascii("mm")); padd(ascii("fo:padding-top"), sXML_CDATA, Double2Str(WTMM(fstyle->margin[1][2])) + ascii("mm")); padd(ascii("fo:padding-bottom"), sXML_CDATA, Double2Str(WTMM(fstyle->margin[1][3])) + ascii("mm")); } if(cell->shade != 0) padd(ascii("fo:background-color"), sXML_CDATA, ascii(hcolor2str( sal::static_int_cast(cell->color), sal::static_int_cast(cell->shade), buf))); } else if( fstyle->boxtype == 'E' ) { padd(ascii("fo:padding"), sXML_CDATA,ascii("0mm")); } else if( fstyle->boxtype == 'L' ) { padd( ascii("style:border-line-width-bottom"), sXML_CDATA, ascii("0.02mm 0.35mm 0.02mm")); padd(ascii("fo:border-bottom"), sXML_CDATA,ascii("0.039cm double #808080")); } else if( fstyle->boxtype == 'G' ) { if( fstyle->margin[1][0] || fstyle->margin[1][1] || fstyle->margin[1][2] || fstyle->margin[1][3] ){ OUString clip = ascii("rect("); clip += Double2Str(WTMM(-fstyle->margin[1][0]) ) + ascii("mm "); clip += Double2Str(WTMM(-fstyle->margin[1][1]) ) + ascii("mm "); clip += Double2Str(WTMM(-fstyle->margin[1][2]) ) + ascii("mm "); clip += Double2Str(WTMM(-fstyle->margin[1][3]) ) + ascii("mm)"); padd(ascii("style:mirror"), sXML_CDATA, ascii("none")); padd(ascii("fo:clip"), sXML_CDATA, clip); } char *cell = (char *)fstyle->cell; padd(ascii("draw:luminance"), sXML_CDATA, ascii(Int2Str(cell[0], "%d%%", buf))); padd(ascii("draw:contrast"), sXML_CDATA, ascii(Int2Str(cell[1], "%d%%", buf))); if( cell[2] == 0 ) padd(ascii("draw:color-mode"), sXML_CDATA, ascii("standard")); else if( cell[2] == 1 ) padd(ascii("draw:color-mode"), sXML_CDATA, ascii("greyscale")); else if( cell[2] == 2 ) padd(ascii("draw:color-mode"), sXML_CDATA, ascii("mono")); } rstartEl(ascii("style:properties"), rList); pList->clear(); rendEl(ascii("style:properties")); rendEl(ascii("style:style")); } char *HwpReader::getTStyleName(int index, char *_buf) { return Int2Str(index, "T%d", _buf); } char *HwpReader::getPStyleName(int index, char *_buf) { return Int2Str(index, "P%d", _buf); } void HwpReader::makeChars(hchar_string & rStr) { rchars(OUString(rStr.c_str())); rStr.clear(); } /** * 문단내에 특수문자가 없고 모든 문자가 동일한 CharShape를 사용하는 경우 */ void HwpReader::make_text_p0(HWPPara * para, bool bParaStart) { hchar_string str; int n; int res; hchar dest[3]; unsigned char firstspace = 0; if( !bParaStart) { padd(ascii("text:style-name"), sXML_CDATA, ascii(getPStyleName(para->GetParaShape().index, buf))); rstartEl(ascii("text:p"), rList); pList->clear(); } if( d->bFirstPara && d->bInBody ) { strcpy(buf,"[문서의 처음]"); /* "Begin of Document" */ padd(ascii("text:name"), sXML_CDATA, OUString(buf, strlen(buf), RTL_TEXTENCODING_UTF8)); rstartEl(ascii("text:bookmark"), rList); pList->clear(); rendEl(ascii("text:bookmark")); d->bFirstPara = false; } if( d->bInHeader ) { makeShowPageNum(); d->bInHeader = false; } padd(ascii("text:style-name"), sXML_CDATA, ascii(getTStyleName(para->cshape.index, buf))); rstartEl(ascii("text:span"), rList); pList->clear(); for (n = 0; n < para->nch && para->hhstr[n]->hh; n += para->hhstr[n]->WSize()) { if (para->hhstr[n]->hh == CH_SPACE && !firstspace) { makeChars(str); rstartEl(ascii("text:s"), rList); rendEl(ascii("text:s")); } else if (para->hhstr[n]->hh == CH_END_PARA) { makeChars(str); rendEl(ascii("text:span")); rendEl(ascii("text:p")); break; } else { if (para->hhstr[n]->hh == CH_SPACE) firstspace = 0; else firstspace = 1; res = hcharconv(para->hhstr[n]->hh, dest, UNICODE); for( int j = 0 ; j < res; j++ ) { str.push_back(dest[j]); } } } } /** * 문단내에 특수문자가 없으나 문자들이 다른 CharShape를 사용하는 경우 */ void HwpReader::make_text_p1(HWPPara * para,bool bParaStart) { hchar_string str; int n; int res; hchar dest[3]; int curr = para->cshape.index; unsigned char firstspace = 0; if( !bParaStart ) { padd(ascii("text:style-name"), sXML_CDATA, ascii(getPStyleName(para->GetParaShape().index, buf))); rstartEl(ascii("text:p"), rList); pList->clear(); } if( d->bFirstPara && d->bInBody ) { /* for HWP's Bookmark */ strcpy(buf,"[문서의 처음]"); /* "Begin of Document" */ padd(ascii("text:name"), sXML_CDATA, OUString(buf, strlen(buf), RTL_TEXTENCODING_UTF8)); rstartEl(ascii("text:bookmark"), rList); pList->clear(); rendEl(ascii("text:bookmark")); d->bFirstPara = false; } if( d->bInHeader ) { makeShowPageNum(); d->bInHeader = false; } padd(ascii("text:style-name"), sXML_CDATA, ascii(getTStyleName(curr, buf))); rstartEl(ascii("text:span"), rList); pList->clear(); for (n = 0; n < para->nch && para->hhstr[n]->hh; n += para->hhstr[n]->WSize()) { if (para->GetCharShape(n)->index != curr) { makeChars(str); rendEl(ascii("text:span")); curr = para->GetCharShape(n)->index; padd(ascii("text:style-name"), sXML_CDATA, ascii(getTStyleName(curr, buf))); rstartEl(ascii("text:span"), rList); pList->clear(); } if (para->hhstr[n]->hh == CH_SPACE && !firstspace) { makeChars(str); rstartEl(ascii("text:s"), rList); rendEl(ascii("text:s")); } else if (para->hhstr[n]->hh == CH_END_PARA) { makeChars(str); rendEl(ascii("text:span")); rendEl(ascii("text:p")); break; } else { if( para->hhstr[n]->hh < CH_SPACE ) continue; if (para->hhstr[n]->hh == CH_SPACE) firstspace = 0; else firstspace = 1; res = hcharconv(para->hhstr[n]->hh, dest, UNICODE); for( int j = 0 ; j < res; j++ ) { str.push_back(dest[j]); } } } } /** * 문단 내의 특수문자가 있으며 문자들이 다른 CharShape를 갖는 경우에 대해 처리 */ void HwpReader::make_text_p3(HWPPara * para,bool bParaStart) { hchar_string str; int n, res; hchar dest[3]; unsigned char firstspace = 0; bool pstart = bParaStart; bool tstart = false; bool infield = false; int curr = para->cshape.index; if( d->bFirstPara && d->bInBody ) { if ( !pstart ) { STARTP; } strcpy(buf,"[문서의 처음]"); /* "Begin of Document" */ padd(ascii("text:name"), sXML_CDATA, OUString(buf, strlen(buf), RTL_TEXTENCODING_UTF8)); rstartEl(ascii("text:bookmark"), rList); pList->clear(); rendEl(ascii("text:bookmark")); d->bFirstPara = false; } if( d->bInHeader ) { if ( !pstart ) { STARTP; } makeShowPageNum(); d->bInHeader = false; } for (n = 0; n < para->nch && para->hhstr[n]->hh; n += para->hhstr[n]->WSize()) { if( para->hhstr[n]->hh == CH_END_PARA ) { if (str.size() > 0) { if( !pstart ){ STARTP;} if( !tstart ){ STARTT;} makeChars(str); } if( tstart ){ ENDT;} if( !pstart ){ STARTP;} if( pstart ){ ENDP;} break; } else if( para->hhstr[n]->hh == CH_SPACE && !firstspace) { if( !pstart ) {STARTP;} if( !tstart ) {STARTT;} makeChars(str); rstartEl(ascii("text:s"), rList); pList->clear(); rendEl(ascii("text:s")); } else if ( para->hhstr[n]->hh >= CH_SPACE ) { if( n > 0 ) if( para->GetCharShape(n)->index != para->GetCharShape(n-1)->index && !infield ) { if( !pstart ) {STARTP;} if( !tstart ) {STARTT;} makeChars(str); ENDT; } if( para->hhstr[n]->hh == CH_SPACE ) firstspace = 0; else firstspace = 1; res = hcharconv(para->hhstr[n]->hh, dest, UNICODE); for( int j = 0 ; j < res; j++ ) { str.push_back(dest[j]); } } else if (para->hhstr[n]->hh == CH_FIELD) { FieldCode *hbox = static_cast(para->hhstr[n]); if( hbox->location_info == 1) { if( !pstart ) {STARTP;} if( !tstart ) {STARTT;} makeChars(str); firstspace = 1; if( hbox->type[0] == 4 && hbox->type[1] == 0 ) { field = hbox->str3; } else{ makeFieldCode(str, hbox); } infield = true; } else { firstspace = 1; if( hbox->type[0] == 4 && hbox->type[1] == 0 ) { makeFieldCode(str, hbox); field = 0L; } infield = false; str.clear(); } } else { switch (para->hhstr[n]->hh) { case CH_BOOKMARK: if( !pstart ) {STARTP;} if( !tstart ) {STARTT;} makeChars(str); makeBookmark(static_cast(para->hhstr[n])); break; case CH_DATE_FORM: // 7 break; case CH_DATE_CODE: // 8 if( !pstart ) {STARTP;} if( !tstart ) {STARTT;} makeChars(str); makeDateCode(static_cast(para->hhstr[n])); break; case CH_TAB: // 9 if( !pstart ) {STARTP;} if (str.size() > 0) { if( !tstart ) {STARTT;} makeChars(str); } makeTab(static_cast(para->hhstr[n])); break; case CH_TEXT_BOX: /* 10 - 표/텍스트박스/수식/버튼/하이퍼텍스트 순 */ { /* 일단은 표만 처리하고, 수식은 text:p안에 들어가는 것으로 처리. */ TxtBox *hbox = static_cast(para->hhstr[n]); if( hbox->style.anchor_type == 0 ) { if( !pstart ) {STARTP;} if( !tstart ) {STARTT;} makeChars(str); } else { if( !pstart ) {STARTP;} if (str.size() > 0) { if( !tstart ) {STARTT;} makeChars(str); } if( tstart ) {ENDT;} } switch (hbox->type) { case TBL_TYPE: // table case TXT_TYPE: // text box case EQU_TYPE: // formula makeTextBox(hbox); break; case BUTTON_TYPE: // text button case HYPERTEXT_TYPE: // hypertext makeHyperText(hbox); break; } break; } case CH_PICTURE: // 11 { Picture *hbox = static_cast(para->hhstr[n]); if( hbox->style.anchor_type == 0 ) { if( !pstart ) {STARTP;} if( !tstart ) {STARTT;} makeChars(str); } else { if( !pstart ) {STARTP;} if (str.size() > 0) { if( !tstart ) {STARTT;} makeChars(str); } if( tstart ) {ENDT;} } makePicture(hbox); break; } case CH_LINE: // 14 { Line *hbox = static_cast(para->hhstr[n]); if (str.size() > 0) { if( !pstart ) {STARTP;} if( !tstart ) {STARTT;} makeChars(str); } if( tstart ) {ENDT;} if( pstart ) {ENDP;} makeLine(hbox); pstart = true; break; } case CH_HIDDEN: // 15 if( !pstart ) {STARTP;} if( !tstart ) {STARTT;} makeChars(str); makeHidden(static_cast(para->hhstr[n])); break; case CH_FOOTNOTE: // 17 if( !pstart ) {STARTP;} if( !tstart ) {STARTT;} makeChars(str); makeFootnote(static_cast(para->hhstr[n])); break; case CH_AUTO_NUM: // 18 if( !pstart ) {STARTP;} if( !tstart ) {STARTT;} makeChars(str); makeAutoNum(static_cast(para->hhstr[n])); break; case CH_NEW_NUM: // 19 -skip break; case CH_PAGE_NUM_CTRL: // 21 break; case CH_MAIL_MERGE: // 22 if( !pstart ) {STARTP;} if( !tstart ) {STARTT;} makeChars(str); makeMailMerge(static_cast(para->hhstr[n])); break; case CH_COMPOSE: /* 23 - 글자겹침 */ break; case CH_HYPHEN: // 24 break; case CH_TOC_MARK: /* 25 아래의 3개는 작업해야 한다. */ if( !pstart ) {STARTP;} if( !tstart ) {STARTT;} makeChars(str); makeTocMark(static_cast(para->hhstr[n])); break; case CH_INDEX_MARK: // 26 if( !pstart ) {STARTP;} if( !tstart ) {STARTT;} makeChars(str); makeIndexMark(static_cast(para->hhstr[n])); break; case CH_OUTLINE: // 28 if( !pstart ) {STARTP;} if( !tstart ) {STARTT;} makeChars(str); makeOutline(static_cast(para->hhstr[n])); break; case CH_FIXED_SPACE: case CH_KEEP_SPACE: str.push_back(0x0020); break; } } } } void HwpReader::makeFieldCode(hchar_string & rStr, FieldCode *hbox) { /* 누름틀 */ if( hbox->type[0] == 4 && hbox->type[1] == 0 ) { padd(ascii("text:placeholder-type"), sXML_CDATA, ascii("text")); if( field ) padd(ascii("text:description"), sXML_CDATA, hconv(field)); rstartEl( ascii("text:placeholder"), rList); pList->clear(); rchars( OUString(rStr.c_str())); rendEl( ascii("text:placeholder") ); } /* 문서요약 */ else if( hbox->type[0] == 3 && hbox->type[1] == 0 ) { if (hconv(hbox->str3) == "title") { rstartEl( ascii("text:title"), rList ); rchars( hconv(hbox->str2) ); rendEl( ascii("text:title") ); } else if (hconv(hbox->str3) == "subject") { rstartEl( ascii("text:subject"), rList ); rchars( hconv(hbox->str2) ); rendEl( ascii("text:subject") ); } else if (hconv(hbox->str3) == "author") { rstartEl( ascii("text:author-name"), rList ); rchars( hconv(hbox->str2) ); rendEl( ascii("text:author-name") ); } else if (hconv(hbox->str3) == "keywords") { rstartEl( ascii("text:keywords"), rList ); rchars( hconv(hbox->str2) ); rendEl( ascii("text:keywords") ); } } /* 개인정보 */ else if( hbox->type[0] == 3 && hbox->type[1] == 1 ) { if (hconv(hbox->str3) == "User") { rstartEl( ascii("text:sender-lastname"), rList ); rchars( hconv(hbox->str2) ); rendEl( ascii("text:sender-lastname") ); } else if (hconv(hbox->str3) == "Company") { rstartEl( ascii("text:sender-company"), rList ); rchars( hconv(hbox->str2) ); rendEl( ascii("text:sender-company") ); } else if (hconv(hbox->str3) == "Position") { rstartEl( ascii("text:sender-title"), rList ); rchars( hconv(hbox->str2) ); rendEl( ascii("text:sender-title") ); } else if (hconv(hbox->str3) == "Division") { rstartEl( ascii("text:sender-position"), rList ); rchars( hconv(hbox->str2) ); rendEl( ascii("text:sender-position") ); } else if (hconv(hbox->str3) == "Fax") { rstartEl( ascii("text:sender-fax"), rList ); rchars( hconv(hbox->str2) ); rendEl( ascii("text:sender-fax") ); } else if (hconv(hbox->str3) == "Pager") { rstartEl( ascii("text:phone-private"), rList ); rchars( hconv(hbox->str2) ); rendEl( ascii("text:phone-private") ); } else if (hconv(hbox->str3) == "E-mail") { rstartEl( ascii("text:sender-email"), rList ); rchars( hconv(hbox->str2) ); rendEl( ascii("text:sender-email") ); } else if (hconv(hbox->str3) == "Zipcode(office)") { rstartEl( ascii("text:sender-postal-code"), rList ); rchars( hconv(hbox->str2) ); rendEl( ascii("text:sender-postal-code") ); } else if (hconv(hbox->str3) == "Phone(office)") { rstartEl( ascii("text:sender-phone-work"), rList ); rchars( hconv(hbox->str2) ); rendEl( ascii("text:sender-phone-work") ); } else if (hconv(hbox->str3) == "Address(office)") { rstartEl( ascii("text:sender-street"), rList ); rchars( hconv(hbox->str2) ); rendEl( ascii("text:sender-street") ); } } else if( hbox->type[0] == 3 && hbox->type[1] == 2 ) /* 만든날짜 */ { if( hbox->m_pDate ) padd(ascii("style:data-style-name"), sXML_CDATA, ascii(Int2Str(hbox->m_pDate->key, "N%d", buf))); rstartEl( ascii("text:creation-date"), rList ); pList->clear(); rchars( hconv(hbox->str2) ); rendEl( ascii("text:creation-date") ); } } /** * Completed * 스타오피스에서는 북마크를 Reference로 참조하나 hwp에는 그 기능이 없다. */ void HwpReader::makeBookmark(Bookmark * hbox) { if (hbox->type == 0) { padd(ascii("text:name"), sXML_CDATA, (hconv(hbox->id))); rstartEl(ascii("text:bookmark"), rList); pList->clear(); rendEl(ascii("text:bookmark")); } else if (hbox->type == 1) /* 블록 북마크일 경우 시작과 끝이 있다 */ { padd(ascii("text:name"), sXML_CDATA, (hconv(hbox->id))); rstartEl(ascii("text:bookmark-start"), rList); pList->clear(); rendEl(ascii("text:bookmark-start")); } else if (hbox->type == 2) { padd(ascii("text:name"), sXML_CDATA, (hconv(hbox->id))); rstartEl(ascii("text:bookmark-end"), rList); pList->clear(); rendEl(ascii("text:bookmark-end")); } } #include "datecode.h" void HwpReader::makeDateFormat(DateCode * hbox) { padd(ascii("style:name"), sXML_CDATA, ascii(Int2Str(hbox->key, "N%d", buf))); padd(ascii("style:family"), sXML_CDATA,ascii("data-style")); padd(ascii("number:language"), sXML_CDATA,ascii("ko")); padd(ascii("number:country"), sXML_CDATA,ascii("KR")); rstartEl(ascii("number:date-style"), rList); pList->clear(); bool add_zero = false; int zero_check = 0; hbox->format[DATE_SIZE -1] = 0; const hchar *fmt = hbox->format[0] ? hbox->format : defaultform; for( ; *fmt ; fmt++ ) { if( zero_check == 1 ) { zero_check = 0; } else add_zero = false; switch( *fmt ) { case '0': zero_check = 1; add_zero = true; break; case '1': padd(ascii("number:style"), sXML_CDATA, ascii("long")); rstartEl(ascii("number:year"), rList); pList->clear(); rendEl(ascii("number:year")); break; case '!': rstartEl(ascii("number:year"), rList); pList->clear(); rendEl(ascii("number:year")); break; case '2': if( add_zero ) padd(ascii("number:style"), sXML_CDATA, ascii("long")); rstartEl(ascii("number:month"), rList); pList->clear(); rendEl(ascii("number:month")); break; case '@': padd(ascii("number:textual"), sXML_CDATA, ascii("true")); rstartEl(ascii("number:month"), rList); pList->clear(); rendEl(ascii("number:month")); break; case '*': padd(ascii("number:textual"), sXML_CDATA, ascii("true")); padd(ascii("number:style"), sXML_CDATA, ascii("long")); rstartEl(ascii("number:month"), rList); pList->clear(); rendEl(ascii("number:month")); break; case '3': if( add_zero ) padd(ascii("number:style"), sXML_CDATA, ascii("long")); rstartEl(ascii("number:day"), rList); pList->clear(); rendEl(ascii("number:day")); break; case '#': if( add_zero ) padd(ascii("number:style"), sXML_CDATA, ascii("long")); rstartEl(ascii("number:day"), rList); pList->clear(); rendEl(ascii("number:day")); switch( hbox->date[DateCode::DAY] % 10) { case 1: rstartEl(ascii("number:text"), rList); rchars(ascii("st")); rendEl(ascii("number:text")); break; case 2: rstartEl(ascii("number:text"), rList); rchars(ascii("nd")); rendEl(ascii("number:text")); break; case 3: rstartEl(ascii("number:text"), rList); rchars(ascii("rd")); rendEl(ascii("number:text")); break; default: rstartEl(ascii("number:text"), rList); rchars(ascii("th")); rendEl(ascii("number:text")); break; } break; case '4': case '$': if( add_zero ) padd(ascii("number:style"), sXML_CDATA, ascii("long")); rstartEl(ascii("number:hours"), rList); pList->clear(); rendEl(ascii("number:hours")); break; case '5': case '%': if( add_zero ) padd(ascii("number:style"), sXML_CDATA, ascii("long")); rstartEl(ascii("number:minutes"), rList); pList->clear(); rendEl(ascii("number:minutes")); break; case '_': padd(ascii("number:style"), sXML_CDATA, ascii("long")); //fall-through case '6': case '^': rstartEl(ascii("number:day-of-week"), rList); pList->clear(); rendEl(ascii("number:day-of-week")); break; case '7': case '&': case '+': rstartEl(ascii("number:am-pm"), rList); pList->clear(); rendEl(ascii("number:am-pm")); break; case '~': // Chinese Locale break; default: hchar sbuf[2]; sbuf[0] = *fmt; sbuf[1] = 0; rstartEl(ascii("number:text"), rList); rchars((hconv(sbuf))); rendEl(ascii("number:text")); break; } } pList->clear(); rendEl(ascii("number:date-style")); } void HwpReader::makeDateCode(DateCode * hbox) { padd(ascii("style:data-style-name"), sXML_CDATA, ascii(Int2Str(hbox->key, "N%d", buf))); rstartEl( ascii("text:date"), rList ); pList->clear(); hchar_string const boxstr = hbox->GetString(); rchars((hconv(boxstr.c_str()))); rendEl( ascii("text:date") ); } void HwpReader::makeTab(Tab * ) /*hbox */ { rstartEl(ascii("text:tab-stop"), rList); rendEl(ascii("text:tab-stop")); } void HwpReader::makeTable(TxtBox * hbox) { padd(ascii("table:name"), sXML_CDATA, ascii(Int2Str(hbox->style.boxnum, "Table%d", buf))); padd(ascii("table:style-name"), sXML_CDATA, ascii(Int2Str(hbox->style.boxnum, "Table%d", buf))); rstartEl(ascii("table:table"), rList); pList->clear(); Table *tbl = hbox->m_pTable; // ----------- column ---------------- for (size_t i = 0 ; i < tbl->columns.nCount -1 ; i++) { sprintf(buf,"Table%d.%c",hbox->style.boxnum, static_cast('A'+i)); padd(ascii("table:style-name"), sXML_CDATA, ascii( buf )); rstartEl(ascii("table:table-column"), rList); pList->clear(); rendEl(ascii("table:table-column")); } // ----------- cell ---------------- int j = -1, k = -1; for (std::list::iterator it = tbl->cells.begin(), aEnd = tbl->cells.end(); it != aEnd; ++it) { TCell *tcell = *it; if( tcell->nRowIndex > j ) { if( j > k ) { rendEl(ascii("table:table-row")); k = j; } // --------------- row ---------------- sprintf(buf,"Table%d.row%d",hbox->style.boxnum, tcell->nRowIndex + 1); padd(ascii("table:style-name"), sXML_CDATA, ascii( buf )); rstartEl(ascii("table:table-row"), rList); pList->clear(); j = tcell->nRowIndex; } sprintf(buf,"Table%d.%c%d",hbox->style.boxnum, 'A'+ tcell->nColumnIndex, tcell->nRowIndex +1); padd(ascii("table:style-name"), sXML_CDATA, ascii( buf )); if( tcell->nColumnSpan > 1 ) padd(ascii("table:number-columns-spanned"), sXML_CDATA, ascii(Int2Str(tcell->nColumnSpan, "%d", buf))); if( tcell->nRowSpan > 1 ) padd(ascii("table:number-rows-spanned"), sXML_CDATA, ascii(Int2Str(tcell->nRowSpan, "%d", buf))); padd(ascii("table:value-type"), sXML_CDATA,ascii("string")); if( tcell->pCell->protect ) padd(ascii("table:protected"), sXML_CDATA,ascii("true")); rstartEl(ascii("table:table-cell"), rList); pList->clear(); parsePara(hbox->plists[tcell->pCell->key].front()); rendEl(ascii("table:table-cell")); } rendEl(ascii("table:table-row")); rendEl(ascii("table:table")); } /** * 텍스트박스와 테이블을 파싱한다. * 1. draw:style-name, draw:name, text:anchor-type, svg:width, * fo:min-height, svg:x, svg:y * TODO : fo:background-color로 셀의 칼라 설정=>스타일에 들어가는 지 아직 모르겠다. */ void HwpReader::makeTextBox(TxtBox * hbox) { if( hbox->style.cap_len > 0 && hbox->type == TXT_TYPE) { padd(ascii("draw:style-name"), sXML_CDATA, ascii(Int2Str(hbox->style.boxnum, "CapBox%d", buf))); padd(ascii("draw:name"), sXML_CDATA, ascii(Int2Str(hbox->style.boxnum, "CaptionBox%d", buf))); padd(ascii("draw:z-index"), sXML_CDATA, ascii(Int2Str(hbox->zorder, "%d", buf))); switch (hbox->style.anchor_type) { case CHAR_ANCHOR: padd(ascii("text:anchor-type"), sXML_CDATA, ascii("as-char")); break; case PARA_ANCHOR: padd(ascii("text:anchor-type"), sXML_CDATA, ascii("paragraph")); break; case PAGE_ANCHOR: case PAPER_ANCHOR: { padd(ascii("text:anchor-type"), sXML_CDATA, ascii("page")); padd(ascii("text:anchor-page-number"), sXML_CDATA, ascii(Int2Str(hbox->pgno +1, "%d", buf))); break; } } if (hbox->style.anchor_type != CHAR_ANCHOR) { padd(ascii("svg:x"), sXML_CDATA, Double2Str(WTMM( ( hbox->pgx + hbox->style.margin[0][0] ) )) + ascii("mm")); padd(ascii("svg:y"), sXML_CDATA, Double2Str(WTMM( ( hbox->pgy + hbox->style.margin[0][2] ) )) + ascii("mm")); } padd(ascii("svg:width"), sXML_CDATA, Double2Str(WTMM(( hbox->box_xs + hbox->cap_xs) )) + ascii("mm")); padd(ascii("fo:min-height"), sXML_CDATA, Double2Str(WTMM(( hbox->box_ys + hbox->cap_ys) )) + ascii("mm")); rstartEl(ascii("draw:text-box"), rList); pList->clear(); if( hbox->cap_pos % 2 ) /* 캡션이 위쪽에 위치한다 */ { parsePara(hbox->caption.front()); } padd( ascii("text:style-name"), sXML_CDATA, ascii("Standard")); rstartEl(ascii("text:p"), rList); pList->clear(); } else{ padd(ascii("draw:z-index"), sXML_CDATA, ascii(Int2Str(hbox->zorder, "%d", buf))); } padd(ascii("draw:style-name"), sXML_CDATA, ascii(Int2Str(hbox->style.boxnum, "Txtbox%d", buf))); padd(ascii("draw:name"), sXML_CDATA, ascii(Int2Str(hbox->style.boxnum, "Frame%d", buf))); if( hbox->style.cap_len <= 0 || hbox->type != TXT_TYPE ) { int x = 0; int y = 0; switch (hbox->style.anchor_type) { case CHAR_ANCHOR: padd(ascii("text:anchor-type"), sXML_CDATA, ascii("as-char")); break; case PARA_ANCHOR: padd(ascii("text:anchor-type"), sXML_CDATA, ascii("paragraph")); break; case PAGE_ANCHOR: case PAPER_ANCHOR: { padd(ascii("text:anchor-type"), sXML_CDATA, ascii("page")); padd(ascii("text:anchor-page-number"), sXML_CDATA, ascii(Int2Str(hbox->pgno +1, "%d", buf))); break; } } if( hbox->style.anchor_type != CHAR_ANCHOR ) { x += hbox->style.margin[0][0]; y += hbox->style.margin[0][2]; } padd(ascii("svg:x"), sXML_CDATA, Double2Str(WTMM( hbox->pgx + x )) + ascii("mm")); padd(ascii("svg:y"), sXML_CDATA, Double2Str(WTMM( hbox->pgy + y )) + ascii("mm")); } else { padd(ascii("text:anchor-type"), sXML_CDATA, ascii("as-char")); padd(ascii("svg:y"), sXML_CDATA, ascii("0cm")); } padd(ascii("svg:width"), sXML_CDATA, Double2Str(WTMM( hbox->box_xs )) + ascii("mm")); if( hbox->style.cap_len > 0 && hbox->type != TXT_TYPE) padd(ascii("fo:min-height"), sXML_CDATA, Double2Str(WTMM( hbox->box_ys + hbox->cap_ys)) + ascii("mm")); else padd(ascii("svg:height"), sXML_CDATA, Double2Str(WTMM(hbox->box_ys )) + ascii("mm")); if( hbox->type != EQU_TYPE ) { rstartEl(ascii("draw:text-box"), rList); pList->clear(); /* 캡션이 존재하고, 위쪽에 있으면 */ if( hbox->style.cap_len > 0 && (hbox->cap_pos % 2) && hbox->type == TBL_TYPE ) { parsePara(hbox->caption.front()); } if( hbox->type == TBL_TYPE) // Is Table { makeTable(hbox); } else // Is TextBox { parsePara(hbox->plists[0].front()); } /* 캡션이 존재하고, 아래쪽에 있으면 */ if( hbox->style.cap_len > 0 && !(hbox->cap_pos % 2) && hbox->type == TBL_TYPE) { parsePara(hbox->caption.front()); } rendEl(ascii("draw:text-box")); // Caption exist and it is text-box if( hbox->style.cap_len > 0 && hbox->type == TXT_TYPE) { rendEl( ascii("text:p")); if( !(hbox->cap_pos % 2)) { parsePara(hbox->caption.front()); } rendEl( ascii("draw:text-box")); } } else // is Formula { rstartEl(ascii("draw:object"), rList); pList->clear(); makeFormula(hbox); rendEl(ascii("draw:object")); } } /** * MathML로 변환해야 한다. * */ void HwpReader::makeFormula(TxtBox * hbox) { char mybuf[3000]; HWPPara* pPar; CharShape *cshape = 0; int n, c, res; hchar dest[3]; size_t l = 0; pPar = hbox->plists[0].front(); while( pPar ) { for( n = 0; n < pPar->nch && pPar->hhstr[n]->hh; n += pPar->hhstr[n]->WSize() ) { if (!cshape) cshape = pPar->GetCharShape(n); if (l >= sizeof(mybuf)-7) break; res = hcharconv(pPar->hhstr[n]->hh, dest, UNICODE); for( int j = 0 ; j < res; j++ ){ c = dest[j]; if( c < 32 ) c = ' '; if( c < 256 ) mybuf[l++] = sal::static_int_cast(c); else { mybuf[l++] = sal::static_int_cast((c >> 8) & 0xff); mybuf[l++] = sal::static_int_cast(c & 0xff); } } } if (l >= sizeof(mybuf)-7) break; mybuf[l++] = '\n'; pPar = pPar->Next(); } mybuf[l] = '\0'; Formula *form = new Formula(mybuf); form->setDocumentHandler(m_rxDocumentHandler); form->setAttributeListImpl(pList); form->parse(); delete form; } /** * platform정보를 읽어들여서 href가 C:\나 D:\로 시작할 경우 리눅스나 솔라리스이면 * C:\ => 홈으로, D:\ => 루트(/)로 바꾸어주는 작업이 필요하다. 이것은 * 한컴이 도스에뮬레이터를 쓰기 때문이다. */ void HwpReader::makeHyperText(TxtBox * hbox) { HyperText *hypert = hwpfile.GetHyperText(); if( !hypert ) return; if( strlen((char *)hypert->filename) > 0 ){ ::std::string const tmp = hstr2ksstr(hypert->bookmark); ::std::string const tmp2 = hstr2ksstr(kstr2hstr( #ifdef _WIN32 (uchar *) urltowin((char *)hypert->filename).c_str()).c_str()); #else (uchar *) urltounix((char *)hypert->filename).c_str()).c_str()); #endif padd(ascii("xlink:type"), sXML_CDATA, ascii("simple")); if (tmp.size() > 0 && strcmp(tmp.c_str(), "[HTML]")) { ::std::string tmp3(tmp2); tmp3.push_back('#'); tmp3.append(tmp); padd(ascii("xlink:href"), sXML_CDATA, OUString(tmp3.c_str(), tmp3.size()+1, RTL_TEXTENCODING_EUC_KR)); } else{ padd(ascii("xlink:href"), sXML_CDATA, OUString(tmp2.c_str(), tmp2.size()+1, RTL_TEXTENCODING_EUC_KR)); } } else { padd(ascii("xlink:type"), sXML_CDATA, ascii("simple")); ::std::string tmp; tmp.push_back('#'); tmp.append(hstr2ksstr(hypert->bookmark)); padd(ascii("xlink:href"), sXML_CDATA, OUString(tmp.c_str(), tmp.size()+1, RTL_TEXTENCODING_EUC_KR)); } rstartEl(ascii("draw:a"), rList); pList->clear(); makeTextBox(hbox); rendEl(ascii("draw:a")); } /** * platform정보를 읽어들여서 href가 C:\나 D:\로 시작할 경우 리눅스나 솔라리스이면 * C:\ => 홈으로, D:\ => 루트(/)로 바꾸었다. 이것은 * 한컴이 도스에뮬레이터를 쓰기 때문이다. */ void HwpReader::makePicture(Picture * hbox) { switch (hbox->pictype) { case PICTYPE_OLE: case PICTYPE_EMBED: case PICTYPE_FILE: { if( hbox->style.cap_len > 0 ) { padd(ascii("draw:style-name"), sXML_CDATA, ascii(Int2Str(hbox->style.boxnum, "CapBox%d", buf))); padd(ascii("draw:name"), sXML_CDATA, ascii(Int2Str(hbox->style.boxnum, "CaptionBox%d", buf))); padd(ascii("draw:z-index"), sXML_CDATA, ascii(Int2Str(hbox->zorder, "%d", buf))); switch (hbox->style.anchor_type) { case CHAR_ANCHOR: padd(ascii("text:anchor-type"), sXML_CDATA, ascii("as-char")); break; case PARA_ANCHOR: padd(ascii("text:anchor-type"), sXML_CDATA, ascii("paragraph")); break; case PAGE_ANCHOR: case PAPER_ANCHOR: { padd(ascii("text:anchor-type"), sXML_CDATA, ascii("page")); padd(ascii("text:anchor-page-number"), sXML_CDATA, ascii(Int2Str(hbox->pgno +1, "%d", buf))); break; } } if (hbox->style.anchor_type != CHAR_ANCHOR) { padd(ascii("svg:x"), sXML_CDATA, Double2Str(WTMM( hbox->pgx + hbox->style.margin[0][0] )) + ascii("mm")); padd(ascii("svg:y"), sXML_CDATA, Double2Str(WTMM( hbox->pgy + hbox->style.margin[0][2] )) + ascii("mm")); } padd(ascii("svg:width"), sXML_CDATA, Double2Str(WTMM( hbox->box_xs + hbox->style.margin[1][0] + hbox->style.margin[1][1] )) + ascii("mm")); padd(ascii("fo:min-height"), sXML_CDATA, Double2Str(WTMM( hbox->box_ys + hbox->style.margin[1][2] + hbox->style.margin[1][3] + hbox->cap_ys )) + ascii("mm")); rstartEl(ascii("draw:text-box"), rList); pList->clear(); if( hbox->cap_pos % 2 ) /* 캡션이 위쪽에 위치한다 */ { parsePara(hbox->caption.front()); } padd( ascii("text:style-name"), sXML_CDATA, ascii("Standard")); rstartEl(ascii("text:p"), rList); pList->clear(); } if( hbox->ishyper ) { padd(ascii("xlink:type"), sXML_CDATA, ascii("simple")); #ifdef _WIN32 if( hbox->follow[4] != 0 ) padd(ascii("xlink:href"), sXML_CDATA, (hconv(kstr2hstr(hbox->follow + 4).c_str()))); else padd(ascii("xlink:href"), sXML_CDATA, (hconv(kstr2hstr(hbox->follow + 5).c_str()))); #else if( hbox->follow[4] != 0 ) padd(ascii("xlink:href"), sXML_CDATA, (hconv(kstr2hstr((uchar *)urltounix((char *)(hbox->follow + 4)).c_str()).c_str()))); else padd(ascii("xlink:href"), sXML_CDATA, (hconv(kstr2hstr((uchar *)urltounix((char *)(hbox->follow + 5)).c_str()).c_str()))); #endif rstartEl(ascii("draw:a"), rList); pList->clear(); } padd(ascii("draw:style-name"), sXML_CDATA, ascii(Int2Str(hbox->style.boxnum, "G%d", buf))); padd(ascii("draw:name"), sXML_CDATA, ascii(Int2Str(hbox->style.boxnum, "Image%d", buf))); if( hbox->style.cap_len <= 0 ) { padd(ascii("draw:z-index"), sXML_CDATA, ascii(Int2Str(hbox->zorder, "%d", buf))); switch (hbox->style.anchor_type) { case CHAR_ANCHOR: padd(ascii("text:anchor-type"), sXML_CDATA, ascii("as-char")); break; case PARA_ANCHOR: padd(ascii("text:anchor-type"), sXML_CDATA, ascii("paragraph")); break; case PAGE_ANCHOR: case PAPER_ANCHOR: { padd(ascii("text:anchor-type"), sXML_CDATA, ascii("page")); padd(ascii("text:anchor-page-number"), sXML_CDATA, ascii(Int2Str(hbox->pgno +1, "%d", buf))); break; } } if (hbox->style.anchor_type != CHAR_ANCHOR) { padd(ascii("svg:x"), sXML_CDATA, Double2Str(WTMM( hbox->pgx + hbox->style.margin[0][0] )) + ascii("mm")); padd(ascii("svg:y"), sXML_CDATA, Double2Str(WTMM( hbox->pgy + hbox->style.margin[0][2] )) + ascii("mm")); } } else { padd(ascii("text:anchor-type"), sXML_CDATA, ascii("as-char")); padd(ascii("svg:y"), sXML_CDATA, ascii("0cm")); } padd(ascii("svg:width"), sXML_CDATA, Double2Str(WTMM( hbox->box_xs + hbox->style.margin[1][0] + hbox->style.margin[1][1])) + ascii("mm")); padd(ascii("svg:height"), sXML_CDATA, Double2Str(WTMM( hbox->box_ys + hbox->style.margin[1][2] + hbox->style.margin[1][3])) + ascii("mm")); if ( hbox->pictype == PICTYPE_FILE ){ #ifdef _WIN32 sprintf(buf, "file:///%s", hbox->picinfo.picun.path ); padd(ascii("xlink:href"), sXML_CDATA, (hconv(kstr2hstr((uchar *) buf).c_str()))); #else padd(ascii("xlink:href"), sXML_CDATA, (hconv(kstr2hstr((uchar *) urltounix(hbox->picinfo.picun.path).c_str()).c_str()))); #endif padd(ascii("xlink:type"), sXML_CDATA, ascii("simple")); padd(ascii("xlink:show"), sXML_CDATA, ascii("embed")); padd(ascii("xlink:actuate"), sXML_CDATA, ascii("onLoad")); } if( hbox->pictype == PICTYPE_OLE ) rstartEl(ascii("draw:object-ole"), rList); else rstartEl(ascii("draw:image"), rList); pList->clear(); if (hbox->pictype == PICTYPE_EMBED || hbox->pictype == PICTYPE_OLE) { rstartEl(ascii("office:binary-data"), rList); pList->clear(); if( hbox->pictype == PICTYPE_EMBED ){ EmPicture *emp = hwpfile.GetEmPicture(hbox); if( emp ) { boost::shared_ptr pStr(base64_encode_string( emp->data, emp->size ), Free()); rchars(ascii(pStr.get())); } } else{ if( hwpfile.oledata ){ #ifdef WIN32 LPSTORAGE srcsto; LPUNKNOWN pObj; wchar_t pathname[200]; MultiByteToWideChar(CP_ACP, 0, hbox->picinfo.picole.embname, -1, pathname, 200); int rc = hwpfile.oledata->pis->OpenStorage(pathname, 0, STGM_READWRITE | STGM_SHARE_EXCLUSIVE | STGM_TRANSACTED, NULL, 0, &srcsto); if (rc != S_OK) { rchars(ascii("")); } else{ rc = OleLoad(srcsto, IID_IUnknown, NULL, (LPVOID*)&pObj); if( rc != S_OK ){ srcsto->Release(); rchars(ascii("")); } else{ boost::shared_ptr pStr(base64_encode_string( (uchar *)pObj, strlen((char *)pObj)), Free()); rchars(ascii(pStr.get())); pObj->Release(); srcsto->Release(); } } #else rchars(ascii("")); #endif } } rendEl(ascii("office:binary-data")); } if( hbox->pictype == PICTYPE_OLE ) rendEl(ascii("draw:object-ole")); else rendEl(ascii("draw:image")); if( hbox->ishyper ) { rendEl(ascii("draw:a")); } if( hbox->style.cap_len > 0 ) { rendEl( ascii("text:p")); if( !(hbox->cap_pos % 2)) /* 캡션이 아래쪽에 위치하면, */ { parsePara(hbox->caption.front()); } rendEl( ascii("draw:text-box")); } break; } case PICTYPE_DRAW: if( hbox->picinfo.picdraw.zorder > 0 ) padd(ascii("draw:z-index"), sXML_CDATA, ascii(Int2Str( hbox->picinfo.picdraw.zorder + 10000, "%d", buf))); makePictureDRAW( (HWPDrawingObject *) hbox->picinfo.picdraw.hdo, hbox); break; case PICTYPE_UNKNOWN: break; } } #define DBL(x) ((x) * (x)) void HwpReader::makePictureDRAW(HWPDrawingObject *drawobj, Picture * hbox) { int x = hbox->pgx; int y = hbox->pgy; int a, b; bool bIsRotate = false; while (drawobj) { padd(ascii("draw:style-name"), sXML_CDATA, ascii(Int2Str(drawobj->index, "Draw%d", buf))); a = 0; b = 0; switch (hbox->style.anchor_type) { case CHAR_ANCHOR: padd(ascii("text:anchor-type"), sXML_CDATA, ascii("as-char")); break; case PARA_ANCHOR: padd(ascii("text:anchor-type"), sXML_CDATA, ascii("paragraph")); break; case PAGE_ANCHOR: case PAPER_ANCHOR: { HWPInfo& hwpinfo = hwpfile.GetHWPInfo(); padd(ascii("text:anchor-type"), sXML_CDATA, ascii("page")); padd(ascii("text:anchor-page-number"), sXML_CDATA, ascii(Int2Str(hbox->pgno +1, "%d", buf))); a = hwpinfo.paper.left_margin; b = hwpinfo.paper.top_margin + hwpinfo.paper.header_length; break; } } if (drawobj->type == HWPDO_CONTAINER) { rstartEl(ascii("draw:g"), rList); pList->clear(); makePictureDRAW(drawobj->child, hbox); rendEl(ascii("draw:g")); } else { bIsRotate = false; if( (drawobj->property.flag & HWPDO_FLAG_ROTATION) && (drawobj->property.parall.pt[0].y != drawobj->property.parall.pt[1].y) && //(drawobj->type == HWPDO_RECT || drawobj->type == HWPDO_ADVANCED_ELLIPSE || drawobj->type == HWPDO_ADVANCED_ARC ) (drawobj->type == HWPDO_RECT || drawobj->type == HWPDO_ADVANCED_ELLIPSE ) ) { int i; ZZParall *pal = &drawobj->property.parall; ZZPoint pt[3], r_pt[3]; for(i = 0 ; i < 3 ; i++ ){ pt[i].x = pal->pt[i].x - drawobj->property.rot_originx; /* 물리좌표계로 변환 */ pt[i].y = -(pal->pt[i].y - drawobj->property.rot_originy); } double rotate, skewX ; /* 2 - 회전각 계산 */ if( pt[1].x == pt[0].x ){ if( pt[1].y > pt[0].y ) rotate = PI/2; else rotate = -(PI/2); } else rotate = atan((double)( pt[1].y - pt[0].y )/(pt[1].x - pt[0].x )); if( pt[1].x < pt[0].x ) rotate += PI; for( i = 0 ; i < 3 ; i++){ r_pt[i].x = (int)(pt[i].x * cos(-(rotate)) - pt[i].y * sin(-(rotate))); r_pt[i].y = (int)(pt[i].y * cos(-(rotate)) + pt[i].x * sin(-(rotate))); } /* 4 - 휜각 계산 */ if( r_pt[2].y == r_pt[1].y ) skewX = 0; else skewX = atan((double)(r_pt[2].x - r_pt[1].x )/( r_pt[2].y - r_pt[1].y )); if( skewX >= PI/2 ) skewX -= PI; if( skewX <= -PI/2 ) skewX += PI; OUString trans; if( skewX != 0.0 && rotate != 0.0 ){ trans = ascii("skewX (") + Double2Str(skewX) + ascii(") rotate (") + Double2Str(rotate) + ascii(") translate (") + Double2Str(WTMM(x + a + drawobj->offset2.x + pal->pt[0].x)) + ascii("mm ") + Double2Str(WTMM(y + b + drawobj->offset2.y + pal->pt[0].y)) + ascii("mm)"); bIsRotate = true; } else if( skewX != 0.0 ){ trans = ascii("skewX (") + Double2Str(skewX) + ascii(") translate (") + Double2Str(WTMM(x + a + drawobj->offset2.x + pal->pt[0].x)) + ascii("mm ") + Double2Str(WTMM(y + b + drawobj->offset2.y + pal->pt[0].y)) + ascii("mm)"); bIsRotate = true; } else if( rotate != 0.0 ){ trans = ascii("rotate (") + Double2Str(rotate) + ascii(") translate (") + Double2Str(WTMM(x + a + drawobj->offset2.x + pal->pt[0].x)) + ascii("mm ") + Double2Str(WTMM(y + b + drawobj->offset2.y + pal->pt[0].y)) + ascii("mm)"); bIsRotate = true; } if( bIsRotate ){ drawobj->extent.w = (int)sqrt(double(DBL(pt[1].x-pt[0].x)+DBL(pt[1].y-pt[0].y))); drawobj->extent.h = (int)sqrt(double(DBL(pt[2].x-pt[1].x)+DBL(pt[2].y-pt[1].y))); padd(ascii("draw:transform"), sXML_CDATA, trans); } } switch (drawobj->type) { case HWPDO_LINE: /* 선 - 시작좌표, 끝좌표. */ if( drawobj->u.line_arc.flip & 0x01 ) { padd(ascii("svg:x1"), sXML_CDATA, Double2Str (WTMM(x + a + drawobj->offset2.x + drawobj->extent.w)) + ascii("mm")); padd(ascii("svg:x2"), sXML_CDATA, Double2Str (WTMM( x + a + drawobj->offset2.x )) + ascii("mm")); } else { padd(ascii("svg:x1"), sXML_CDATA, Double2Str (WTMM( x + a + drawobj->offset2.x )) + ascii("mm")); padd(ascii("svg:x2"), sXML_CDATA, Double2Str (WTMM(x + a + drawobj->offset2.x + drawobj->extent.w)) + ascii("mm")); } if( drawobj->u.line_arc.flip & 0x02 ) { padd(ascii("svg:y1"), sXML_CDATA, Double2Str (WTMM( y + b + drawobj->offset2.y + drawobj->extent.h ) ) + ascii("mm")); padd(ascii("svg:y2"), sXML_CDATA, Double2Str (WTMM( y + b + drawobj->offset2.y )) + ascii("mm")); } else { padd(ascii("svg:y1"), sXML_CDATA, Double2Str (WTMM( y + b + drawobj->offset2.y)) + ascii("mm")); padd(ascii("svg:y2"), sXML_CDATA, Double2Str (WTMM(y + b + drawobj->offset2.y + drawobj->extent.h)) + ascii("mm")); } rstartEl(ascii("draw:line"), rList); pList->clear(); rendEl(ascii("draw:line")); break; case HWPDO_RECT: /* 사각형 - 시작위치, 가로/세로 */ if( !bIsRotate ) { padd(ascii("svg:x"), sXML_CDATA, Double2Str (WTMM( x + a + drawobj->offset2.x)) + ascii("mm")); padd(ascii("svg:y"), sXML_CDATA, Double2Str (WTMM( y + b + drawobj->offset2.y)) + ascii("mm")); } padd(ascii("svg:width"), sXML_CDATA, Double2Str (WTMM( drawobj->extent.w )) + ascii("mm")); padd(ascii("svg:height"), sXML_CDATA, Double2Str (WTMM( drawobj->extent.h )) + ascii("mm")); if( drawobj->property.flag & 0x01 ) { int value = drawobj->extent.w < drawobj->extent.h ? drawobj->extent.w : drawobj->extent.h ; padd(ascii("draw:corner-radius"), sXML_CDATA, Double2Str (WTMM( value/10 )) + ascii("mm")); } else if( drawobj->property.flag & 0x04 ) { int value = drawobj->extent.w < drawobj->extent.h ? drawobj->extent.w : drawobj->extent.h ; padd(ascii("draw:corner-radius"), sXML_CDATA, Double2Str (WTMM( value / 2)) + ascii("mm")); } rstartEl(ascii("draw:rect"), rList); pList->clear(); if( (drawobj->property.flag & HWPDO_FLAG_AS_TEXTBOX) && drawobj->property.pPara ) // As Textbox { HWPPara *pPara = drawobj->property.pPara; //parsePara(pPara); while(pPara) { make_text_p1( pPara ); pPara = pPara->Next(); } } rendEl(ascii("draw:rect")); break; case HWPDO_ELLIPSE: /* 타원 - 시작위치, 가로/세로 */ case HWPDO_ADVANCED_ELLIPSE: /* 변형된 타원 */ { if( !bIsRotate ) { padd(ascii("svg:x"), sXML_CDATA, Double2Str (WTMM( x + a + drawobj->offset2.x)) + ascii("mm")); padd(ascii("svg:y"), sXML_CDATA, Double2Str (WTMM( y + b + drawobj->offset2.y)) + ascii("mm")); } padd(ascii("svg:width"), sXML_CDATA, Double2Str (WTMM( drawobj->extent.w )) + ascii("mm")); padd(ascii("svg:height"), sXML_CDATA, Double2Str (WTMM( drawobj->extent.h )) + ascii("mm")); if( drawobj->type == HWPDO_ADVANCED_ELLIPSE ){ if( drawobj->u.arc.radial[0].x != drawobj->u.arc.radial[1].x || drawobj->u.arc.radial[0].y != drawobj->u.arc.radial[1].y ){ int Cx,Cy; Cx = ( drawobj->offset2.x + drawobj->extent.w ) / 2; Cy = ( drawobj->offset2.y + drawobj->extent.h ) / 2; double start_angle, end_angle; start_angle = calcAngle( Cx, Cy, drawobj->u.arc.radial[0].x, drawobj->u.arc.radial[0].y ); end_angle = calcAngle( Cx, Cy, drawobj->u.arc.radial[1].x, drawobj->u.arc.radial[1].y ); if( drawobj->property.fill_color < 0xffffff ) padd(ascii("draw:kind"), sXML_CDATA, ascii("section")); else padd(ascii("draw:kind"), sXML_CDATA, ascii("arc")); padd(ascii("draw:start-angle"), sXML_CDATA, Double2Str(start_angle )); padd(ascii("draw:end-angle"), sXML_CDATA, Double2Str(end_angle)); } } rstartEl(ascii("draw:ellipse"), rList); pList->clear(); if( drawobj->property.flag >> 19 & 0x01 && drawobj->property.pPara ) // As Textbox { HWPPara *pPara = drawobj->property.pPara; //parsePara(pPara); while(pPara) { make_text_p1( pPara ); pPara = pPara->Next(); } } rendEl(ascii("draw:ellipse")); break; } case HWPDO_ARC: /* 호 */ case HWPDO_ADVANCED_ARC: { /* 호일경우에, 스타오피스는 전체 타원의 크기를 사이즈로 한다. */ uint flip = drawobj->u.line_arc.flip; if( !bIsRotate ) { if( ( flip == 0 || flip == 2 ) && drawobj->type == HWPDO_ARC) padd(ascii("svg:x"), sXML_CDATA, Double2Str (WTMM( x + a + drawobj->offset2.x - drawobj->extent.w)) + ascii("mm")); else padd(ascii("svg:x"), sXML_CDATA, Double2Str (WTMM( x + a + drawobj->offset2.x)) + ascii("mm")); if( ( flip == 0 || flip == 1 ) && drawobj->type == HWPDO_ARC) padd(ascii("svg:y"), sXML_CDATA, Double2Str (WTMM( y + b + drawobj->offset2.y - drawobj->extent.h)) + ascii("mm")); else padd(ascii("svg:y"), sXML_CDATA, Double2Str (WTMM( y + b + drawobj->offset2.y)) + ascii("mm")); } padd(ascii("svg:width"), sXML_CDATA, Double2Str (WTMM( drawobj->extent.w * 2)) + ascii("mm")); padd(ascii("svg:height"), sXML_CDATA, Double2Str (WTMM( drawobj->extent.h * 2)) + ascii("mm")); if( drawobj->property.flag & HWPDO_FLAG_DRAW_PIE || drawobj->property.fill_color < 0xffffff ) padd(ascii("draw:kind"), sXML_CDATA, ascii("section")); else padd(ascii("draw:kind"), sXML_CDATA, ascii("arc")); if( drawobj->type == HWPDO_ADVANCED_ARC ){ double start_angle, end_angle; ZZParall *pal = &drawobj->property.parall; if( pal->pt[1].x == pal->pt[0].x ){ if( pal->pt[0].y < pal->pt[1].y ) start_angle = 1.5 * PI; else start_angle = 0.5 * PI; } else{ start_angle = atan((double)( pal->pt[0].y - pal->pt[1].y )/( pal->pt[1].x - pal->pt[0].x )); if( pal->pt[1].x < pal->pt[0].x ) start_angle += PI; } if( pal->pt[1].x == pal->pt[2].x ){ if( pal->pt[2].y < pal->pt[1].y ) end_angle = 1.5 * PI; else end_angle = 0.5 * PI; } else{ end_angle = atan((double)( pal->pt[2].y - pal->pt[1].y )/( pal->pt[1].x - pal->pt[2].x )); if( pal->pt[1].x < pal->pt[2].x ) end_angle += PI; } if( start_angle >= 2 * PI ) start_angle -= 2 * PI; if( end_angle >= 2 * PI ) end_angle -= 2 * PI; if( ( start_angle > end_angle ) && (start_angle - end_angle < PI )){ double tmp_angle = start_angle; start_angle = end_angle; end_angle = tmp_angle; } padd(ascii("draw:start-angle"), sXML_CDATA, Double2Str(start_angle * 180. / PI)); padd(ascii("draw:end-angle"), sXML_CDATA, Double2Str(end_angle * 180. / PI)); } else{ if( drawobj->u.line_arc.flip == 0 ) { padd(ascii("draw:start-angle"), sXML_CDATA, ascii("270")); padd(ascii("draw:end-angle"), sXML_CDATA, ascii("0")); } else if( drawobj->u.line_arc.flip == 1 ) { padd(ascii("draw:start-angle"), sXML_CDATA, ascii("180")); padd(ascii("draw:end-angle"), sXML_CDATA, ascii("270")); } else if( drawobj->u.line_arc.flip == 2 ) { padd(ascii("draw:start-angle"), sXML_CDATA, ascii("0")); padd(ascii("draw:end-angle"), sXML_CDATA, ascii("90")); } else { padd(ascii("draw:start-angle"), sXML_CDATA, ascii("90")); padd(ascii("draw:end-angle"), sXML_CDATA, ascii("180")); } } rstartEl(ascii("draw:ellipse"), rList); pList->clear(); if( drawobj->property.flag >> 19 & 0x01 && drawobj->property.pPara ) // As Textbox { HWPPara *pPara = drawobj->property.pPara; //parsePara(pPara); while(pPara) { make_text_p1( pPara ); pPara = pPara->Next(); } } rendEl(ascii("draw:ellipse")); break; } case HWPDO_CURVE: /* 곡선 : 다각형으로 변환. */ { bool bIsNatural = true; if( drawobj->property.flag >> 5 & 0x01){ bIsNatural = false; } if( !bIsRotate ) { padd(ascii("svg:x"), sXML_CDATA, Double2Str (WTMM( x + a + drawobj->offset2.x)) + ascii("mm")); padd(ascii("svg:y"), sXML_CDATA, Double2Str (WTMM( y + b + drawobj->offset2.y)) + ascii("mm")); } padd(ascii("svg:width"), sXML_CDATA, Double2Str (WTMM( drawobj->extent.w )) + ascii("mm")); padd(ascii("svg:height"), sXML_CDATA, Double2Str (WTMM( drawobj->extent.h )) + ascii("mm")); sprintf(buf, "0 0 %d %d", WTSM(drawobj->extent.w) , WTSM(drawobj->extent.h) ); padd(ascii("svg:viewBox"), sXML_CDATA, ascii(buf) ); OUString oustr; if ((drawobj->u.freeform.npt > 2) && (static_cast(drawobj->u.freeform.npt) < ((::std::numeric_limits::max)() / sizeof(double)))) { int n, i; n = drawobj->u.freeform.npt; double *xarr = new double[n+1]; double *yarr = new double[n+1]; double *tarr = new double[n+1]; double *xb = 0L; double *yb = 0L; double *carr = 0L; double *darr = 0L; for( i = 0 ; i < n ; i++ ){ xarr[i] = drawobj->u.freeform.pt[i].x; yarr[i] = drawobj->u.freeform.pt[i].y; tarr[i] = i; } xarr[n] = xarr[0]; yarr[n] = yarr[0]; tarr[n] = n; if( bIsNatural == false ){ PeriodicSpline(n, tarr, xarr, xb, carr, darr); // prevent memory leak delete[] carr; carr = 0; delete[] darr; darr = 0; PeriodicSpline(n, tarr, yarr, yb, carr, darr); } else{ NaturalSpline(n, tarr, xarr, xb, carr, darr); // prevent memory leak delete[] carr; carr = 0; delete[] darr; darr = 0; NaturalSpline(n, tarr, yarr, yb, carr, darr); } sprintf(buf, "M%d %dC%d %d", WTSM((int)xarr[0]), WTSM((int)yarr[0]), WTSM((int)(xarr[0] + xb[0]/3)), WTSM((int)(yarr[0] + yb[0]/3)) ); oustr += ascii(buf); for( i = 1 ; i < n ; i++ ){ if( i == n -1 ){ sprintf(buf, " %d %d %d %dz", WTSM((int)(xarr[i] - xb[i]/3)), WTSM((int)(yarr[i] - yb[i]/3)), WTSM((int)xarr[i]), WTSM((int)yarr[i]) ); } else{ sprintf(buf, " %d %d %d %d %d %d", WTSM((int)(xarr[i] - xb[i]/3)), WTSM((int)(yarr[i] - yb[i]/3)), WTSM((int)xarr[i]), WTSM((int)yarr[i]), WTSM((int)xarr[i] + xb[i]/3), WTSM((int)(yarr[i] + yb[i]/3)) ); } oustr += ascii(buf); } delete[] tarr; delete[] xarr; delete[] yarr; delete[] xb; delete[] yb; delete[] carr; delete[] darr; } padd(ascii("svg:d"), sXML_CDATA, oustr); rstartEl(ascii("draw:path"), rList); pList->clear(); // As Textbox if( drawobj->property.flag >> 19 & 0x01 && drawobj->property.pPara ) { HWPPara *pPara = drawobj->property.pPara; while(pPara) { make_text_p1( pPara ); pPara = pPara->Next(); } } rendEl(ascii("draw:path")); break; } case HWPDO_CLOSED_FREEFORM: case HWPDO_FREEFORM: /* 다각형 */ { bool bIsPolygon = false; padd(ascii("svg:x"), sXML_CDATA, Double2Str (WTMM( x + a + drawobj->offset2.x)) + ascii("mm")); padd(ascii("svg:y"), sXML_CDATA, Double2Str (WTMM( y + b + drawobj->offset2.y)) + ascii("mm")); padd(ascii("svg:width"), sXML_CDATA, Double2Str (WTMM( drawobj->extent.w )) + ascii("mm")); padd(ascii("svg:height"), sXML_CDATA, Double2Str (WTMM( drawobj->extent.h )) + ascii("mm")); sprintf(buf, "0 0 %d %d", WTSM(drawobj->extent.w), WTSM(drawobj->extent.h)); padd(ascii("svg:viewBox"), sXML_CDATA, ascii(buf) ); OUString oustr; if (drawobj->u.freeform.npt > 0) { sprintf(buf, "%d,%d", WTSM(drawobj->u.freeform.pt[0].x), WTSM(drawobj->u.freeform.pt[0].y)); oustr += ascii(buf); int i; for (i = 1; i < drawobj->u.freeform.npt ; i++) { sprintf(buf, " %d,%d", WTSM(drawobj->u.freeform.pt[i].x), WTSM(drawobj->u.freeform.pt[i].y)); oustr += ascii(buf); } if( drawobj->u.freeform.pt[0].x == drawobj->u.freeform.pt[i-1].x && drawobj->u.freeform.pt[0].y == drawobj->u.freeform.pt[i-1].y ) { bIsPolygon = true; } } padd(ascii("draw:points"), sXML_CDATA, oustr); if( drawobj->property.fill_color <= 0xffffff || drawobj->property.pattern_type != 0) { bIsPolygon = true; } if(bIsPolygon) { rstartEl(ascii("draw:polygon"), rList); pList->clear(); if( drawobj->property.flag >> 19 & 0x01 && // As Textbox drawobj->property.pPara ) { HWPPara *pPara = drawobj->property.pPara; // parsePara(pPara); while(pPara) { make_text_p1( pPara ); pPara = pPara->Next(); } } rendEl(ascii("draw:polygon")); } else { rstartEl(ascii("draw:polyline"), rList); pList->clear(); if( drawobj->property.flag >> 19 & 0x01 && // As Textbox drawobj->property.pPara ) { HWPPara *pPara = drawobj->property.pPara; //parsePara(pPara); while(pPara) { make_text_p1( pPara ); pPara = pPara->Next(); } } rendEl(ascii("draw:polyline")); } break; } case HWPDO_TEXTBOX: if( !bIsRotate ) { padd(ascii("svg:x"), sXML_CDATA, Double2Str (WTMM( x + a + drawobj->offset2.x)) + ascii("mm")); padd(ascii("svg:y"), sXML_CDATA, Double2Str (WTMM( y + b + drawobj->offset2.y)) + ascii("mm")); } padd(ascii("svg:width"), sXML_CDATA, Double2Str (WTMM( drawobj->extent.w )) + ascii("mm")); padd(ascii("svg:height"), sXML_CDATA, Double2Str (WTMM( drawobj->extent.h )) + ascii("mm")); if( drawobj->property.flag & 0x01 ) { int value = drawobj->extent.w < drawobj->extent.h ? drawobj->extent.w : drawobj->extent.h ; padd(ascii("draw:corner-radius"), sXML_CDATA, Double2Str (WTMM( value/10 )) + ascii("mm")); } else if( drawobj->property.flag & 0x04 ) { int value = drawobj->extent.w < drawobj->extent.h ? drawobj->extent.w : drawobj->extent.h ; padd(ascii("draw:corner-radius"), sXML_CDATA, Double2Str (WTMM( value / 2)) + ascii("mm")); } rstartEl(ascii("draw:text-box"), rList); pList->clear(); HWPPara *pPara = drawobj->u.textbox.h; //parsePara(pPara); while(pPara) { make_text_p1( pPara ); pPara = pPara->Next(); } rendEl(ascii("draw:text-box")); break; } } pList->clear(); drawobj = drawobj->next; } } /** * */ void HwpReader::makeLine(Line * ) { padd(ascii("text:style-name"), sXML_CDATA, ascii("Horizontal Line")); rstartEl( ascii("text:p"), rList); pList->clear(); } /** * 입력-주석-숨은설명 : 사용자에게 숨은 설명을 보여준다. * 문단이 포함될 수 있으나, 단지 문자열만 뽑아내어 파싱한다. */ void HwpReader::makeHidden(Hidden * hbox) { hchar_string str; int res; hchar dest[3]; padd(ascii("text:condition"), sXML_CDATA, ascii("")); padd(ascii("text:string-value"), sXML_CDATA, ascii("")); rstartEl(ascii("text:hidden-text"), rList); pList->clear(); HWPPara *para = hbox->plist.front(); while (para) { for (int n = 0; n < para->nch && para->hhstr[n]->hh; n += para->hhstr[n]->WSize()) { res = hcharconv(para->hhstr[n]->hh, dest, UNICODE); for( int j = 0 ; j < res ; j++ ) { str.push_back(dest[j]); } } para = para->Next(); } makeChars(str); rendEl(ascii("text:hidden-text")); } /** * 각주는 text:footnote, 미주는 text:endnote로 변환 */ void HwpReader::makeFootnote(Footnote * hbox) { if (hbox->type) { padd(ascii("text:id"), sXML_CDATA, ascii(Int2Str(hbox->number, "edn%d", buf))); rstartEl(ascii("text:endnote"), rList); pList->clear(); padd(ascii("text:label"), sXML_CDATA, ascii(Int2Str(hbox->number, "%d", buf))); rstartEl(ascii("text:endnote-citation"), rList); pList->clear(); rchars(ascii(Int2Str(hbox->number, "%d", buf))); rendEl(ascii("text:endnote-citation")); rstartEl(ascii("text:endnote-body"), rList); parsePara(hbox->plist.front()); rendEl(ascii("text:endnote-body")); rendEl(ascii("text:endnote")); } else { padd(ascii("text:id"), sXML_CDATA, ascii(Int2Str(hbox->number, "ftn%d", buf))); rstartEl(ascii("text:footnote"), rList); pList->clear(); padd(ascii("text:label"), sXML_CDATA, ascii(Int2Str(hbox->number, "%d", buf))); rstartEl(ascii("text:footnote-citation"), rList); pList->clear(); rchars(ascii(Int2Str(hbox->number, "%d", buf))); rendEl(ascii("text:footnote-citation")); rstartEl(ascii("text:footnote-body"), rList); parsePara(hbox->plist.front()); rendEl(ascii("text:footnote-body")); rendEl(ascii("text:footnote")); } } /** * page/footnote/endnote/picture/table/formula number */ void HwpReader::makeAutoNum(AutoNum * hbox) { switch (hbox->type) { case PGNUM_AUTO: rstartEl(ascii("text:page-number"), rList); rchars(ascii(Int2Str(hbox->number, "%d", buf))); rendEl(ascii("text:page-number")); break; case FNNUM_AUTO: break; case ENNUM_AUTO: break; case EQUNUM_AUTO: case PICNUM_AUTO: padd(ascii("text:ref-name"),sXML_CDATA, ascii(Int2Str(hbox->number, "refIllustration%d", buf))); padd(ascii("text:name"),sXML_CDATA, ascii("Illustration")); padd(ascii("style:num-format"),sXML_CDATA, ascii("1")); rstartEl(ascii("text:sequence"), rList); rchars(ascii(Int2Str(hbox->number, "%d", buf))); rendEl(ascii("text:sequence")); break; case TBLNUM_AUTO: padd(ascii("text:ref-name"),sXML_CDATA, ascii(Int2Str(hbox->number, "refTable%d", buf))); padd(ascii("text:name"),sXML_CDATA, ascii("Table")); padd(ascii("style:num-format"),sXML_CDATA, ascii("1")); rstartEl(ascii("text:sequence"), rList); rchars(ascii(Int2Str(hbox->number, "%d", buf))); rendEl(ascii("text:sequence")); break; } } void HwpReader::makeShowPageNum() { ShowPageNum *hbox = d->pPn; int nPos = 0; if( hbox->where == 1 || hbox->where == 4 ) nPos = 1; else if( hbox->where == 2 || hbox->where == 5 ) nPos = 2; else if( hbox->where == 3 || hbox->where == 6 ) nPos = 3; else /* 이 경우가 존재하면 안된다. */ { if( d->nPnPos == 1 ) nPos = 1; else if( d->nPnPos == 3 ) nPos = 3; } padd(ascii("draw:style-name"), sXML_CDATA, ascii(Int2Str(nPos, "PNBox%d", buf))); padd(ascii("draw:name"), sXML_CDATA, ascii(Int2Str(nPos, "PageNumber%d", buf))); padd(ascii("text:anchor-type"), sXML_CDATA, ascii("paragraph")); padd(ascii("svg:y"), sXML_CDATA, ascii("0cm")); padd(ascii("svg:width"), sXML_CDATA, ascii("2.0cm")); padd(ascii("fo:min-height"), sXML_CDATA, ascii("0.5cm")); rstartEl(ascii("draw:text-box"), rList); pList->clear(); padd(ascii("text:style-name"), sXML_CDATA, ascii(Int2Str(nPos, "PNPara%d", buf))); rstartEl(ascii("text:p"), rList); pList->clear(); if( hbox->shape > 2 ) rchars(ascii("- ")); if( hbox->shape % 3 == 0 ) padd(ascii("style:num-format"), sXML_CDATA, ascii("1")); else if( hbox->shape % 3 == 1 ) padd(ascii("style:num-format"), sXML_CDATA, ascii("I")); else padd(ascii("style:num-format"), sXML_CDATA, ascii("i")); padd(ascii("text:select-page"), sXML_CDATA, ascii("current")); rstartEl(ascii("text:page-number"), rList); pList->clear(); rchars(ascii("2")); rendEl(ascii("text:page-number")); if( hbox->shape > 2 ) rchars(ascii(" -")); rendEl(ascii("text:p")); rendEl(ascii("draw:text-box")); } /** * mail merge operation using hwp addressbook and hwp data form. * not support operation in OO writer. */ void HwpReader::makeMailMerge(MailMerge * hbox) { hchar_string const boxstr = hbox->GetString(); rchars((hconv(boxstr.c_str()))); } /** * Make heading contents file using toc marks * not support operation. */ void HwpReader::makeTocMark(TocMark * ) /*hbox */ { } /** * Make search character table in automatic * not support operation */ void HwpReader::makeIndexMark(IndexMark * ) /*hbox */ { } void HwpReader::makeOutline(Outline * hbox) { if( hbox->kind == 1 ) rchars(OUString(hbox->GetUnicode().c_str())); } void HwpReader::parsePara(HWPPara * para, bool bParaStart) { while (para) { if( para->nch == 1) { if( !bParaStart ) { padd(ascii("text:style-name"), sXML_CDATA, ascii(getPStyleName(para->GetParaShape().index, buf))); rstartEl( ascii("text:p"),rList); pList->clear(); } if( d->bFirstPara && d->bInBody ) { /* for HWP's Bookmark */ strcpy(buf,"[문서의 처음]"); /* "Begin of Document" */ padd(ascii("text:name"), sXML_CDATA, OUString(buf, strlen(buf), RTL_TEXTENCODING_UTF8)); rstartEl(ascii("text:bookmark"), rList); pList->clear(); rendEl(ascii("text:bookmark")); d->bFirstPara = false; } if( d->bInHeader ) { makeShowPageNum(); d->bInHeader = false; } rendEl( ascii("text:p") ); } else { if (!para->ctrlflag) { if (para->contain_cshape) make_text_p1(para, bParaStart); else make_text_p0(para, bParaStart); } else make_text_p3(para, bParaStart); } bParaStart = false; para = para->Next(); } } /* vim:set shiftwidth=4 softtabstop=4 expandtab: */