1 ///////////////////////////////////////////////////////////////////////////// 
   2 // Name:        src/richtext/richtextxml.cpp 
   3 // Purpose:     XML and HTML I/O for wxRichTextCtrl 
   4 // Author:      Julian Smart 
   8 // Copyright:   (c) Julian Smart 
   9 // Licence:     wxWindows licence 
  10 ///////////////////////////////////////////////////////////////////////////// 
  12 // For compilers that support precompilation, includes "wx.h". 
  13 #include "wx/wxprec.h" 
  19 #if wxUSE_RICHTEXT && wxUSE_XML 
  21 #include "wx/richtext/richtextxml.h" 
  25     #include "wx/module.h" 
  28 #include "wx/filename.h" 
  29 #include "wx/clipbrd.h" 
  30 #include "wx/wfstream.h" 
  31 #include "wx/sstream.h" 
  32 #include "wx/txtstrm.h" 
  33 #include "wx/tokenzr.h" 
  34 #include "wx/xml/xml.h" 
  36 IMPLEMENT_DYNAMIC_CLASS(wxRichTextXMLHandler
, wxRichTextFileHandler
) 
  39 bool wxRichTextXMLHandler::DoLoadFile(wxRichTextBuffer 
*buffer
, wxInputStream
& stream
) 
  44     buffer
->ResetAndClearCommands(); 
  47     wxXmlDocument
* xmlDoc 
= new wxXmlDocument
; 
  50     // This is the encoding to convert to (memory encoding rather than file encoding) 
  51     wxString 
encoding(wxT("UTF-8")); 
  53 #if !wxUSE_UNICODE && wxUSE_INTL 
  54     encoding 
= wxLocale::GetSystemEncodingName(); 
  57     if (!xmlDoc
->Load(stream
, encoding
)) 
  59         buffer
->ResetAndClearCommands(); 
  64         if (xmlDoc
->GetRoot() && xmlDoc
->GetRoot()->GetType() == wxXML_ELEMENT_NODE 
&& xmlDoc
->GetRoot()->GetName() == wxT("richtext")) 
  66             wxXmlNode
* child 
= xmlDoc
->GetRoot()->GetChildren(); 
  69                 if (child
->GetType() == wxXML_ELEMENT_NODE
) 
  71                     wxString name 
= child
->GetName(); 
  72                     if (name 
== wxT("richtext-version")) 
  76                         ImportXML(buffer
, child
); 
  79                 child 
= child
->GetNext(); 
  90     buffer
->UpdateRanges(); 
  95 /// Recursively import an object 
  96 bool wxRichTextXMLHandler::ImportXML(wxRichTextBuffer
* buffer
, wxXmlNode
* node
) 
  98     wxString name 
= node
->GetName(); 
 100     bool doneChildren 
= false; 
 102     if (name 
== wxT("paragraphlayout")) 
 104         wxString partial 
= node
->GetPropVal(wxT("partialparagraph"), wxEmptyString
); 
 105         if (partial 
== wxT("true")) 
 106             buffer
->SetPartialParagraph(true); 
 108     else if (name 
== wxT("paragraph")) 
 110         wxRichTextParagraph
* para 
= new wxRichTextParagraph(buffer
); 
 111         buffer
->AppendChild(para
); 
 113         GetStyle(para
->GetAttributes(), node
, true); 
 115         wxXmlNode
* child 
= node
->GetChildren(); 
 118             wxString childName 
= child
->GetName(); 
 119             if (childName 
== wxT("text")) 
 122                 wxXmlNode
* textChild 
= child
->GetChildren(); 
 125                     if (textChild
->GetType() == wxXML_TEXT_NODE 
|| 
 126                         textChild
->GetType() == wxXML_CDATA_SECTION_NODE
) 
 128                         wxString text2 
= textChild
->GetContent(); 
 130                         // Strip whitespace from end 
 131                         if (!text2
.empty() && text2
[text2
.length()-1] == wxT('\n')) 
 132                             text2 
= text2
.Mid(0, text2
.length()-1); 
 134                         if (!text2
.empty() && text2
[0] == wxT('"')) 
 135                             text2 
= text2
.Mid(1); 
 136                         if (!text2
.empty() && text2
[text2
.length()-1] == wxT('"')) 
 137                             text2 
= text2
.Mid(0, text2
.length() - 1); 
 141                     textChild 
= textChild
->GetNext(); 
 144                 wxRichTextPlainText
* textObject 
= new wxRichTextPlainText(text
, para
); 
 145                 GetStyle(textObject
->GetAttributes(), child
, false); 
 147                 para
->AppendChild(textObject
); 
 149             else if (childName 
== wxT("symbol")) 
 151                 // This is a symbol that XML can't read in the normal way 
 153                 wxXmlNode
* textChild 
= child
->GetChildren(); 
 156                     if (textChild
->GetType() == wxXML_TEXT_NODE 
|| 
 157                         textChild
->GetType() == wxXML_CDATA_SECTION_NODE
) 
 159                         wxString text2 
= textChild
->GetContent(); 
 162                     textChild 
= textChild
->GetNext(); 
 166                 actualText 
<< (wxChar
) wxAtoi(text
); 
 168                 wxRichTextPlainText
* textObject 
= new wxRichTextPlainText(actualText
, para
); 
 169                 GetStyle(textObject
->GetAttributes(), child
, false); 
 171                 para
->AppendChild(textObject
); 
 173             else if (childName 
== wxT("image")) 
 175                 int imageType 
= wxBITMAP_TYPE_PNG
; 
 176                 wxString value 
= node
->GetPropVal(wxT("imagetype"), wxEmptyString
); 
 178                     imageType 
= wxAtoi(value
); 
 182                 wxXmlNode
* imageChild 
= child
->GetChildren(); 
 185                     wxString childName 
= imageChild
->GetName(); 
 186                     if (childName 
== wxT("data")) 
 188                         wxXmlNode
* dataChild 
= imageChild
->GetChildren(); 
 191                             data 
= dataChild
->GetContent(); 
 193                             dataChild 
= dataChild
->GetNext(); 
 197                     imageChild 
= imageChild
->GetNext(); 
 202                     wxRichTextImage
* imageObj 
= new wxRichTextImage(para
); 
 203                     para
->AppendChild(imageObj
); 
 205                     wxStringInputStream 
strStream(data
); 
 207                     imageObj
->GetImageBlock().ReadHex(strStream
, data
.length(), imageType
); 
 210             child 
= child
->GetNext(); 
 215     else if (name 
== wxT("stylesheet")) 
 217         if (GetFlags() & wxRICHTEXT_HANDLER_INCLUDE_STYLESHEET
) 
 219             wxRichTextStyleSheet
* sheet 
= new wxRichTextStyleSheet
; 
 220             wxString sheetName 
= node
->GetPropVal(wxT("name"), wxEmptyString
); 
 221             wxString sheetDescription 
= node
->GetPropVal(wxT("description"), wxEmptyString
); 
 222             sheet
->SetName(sheetName
); 
 223             sheet
->SetDescription(sheetDescription
); 
 225             wxXmlNode
* child 
= node
->GetChildren(); 
 228                 ImportStyleDefinition(sheet
, child
); 
 230                 child 
= child
->GetNext(); 
 233             // Notify that styles have changed. If this is vetoed by the app, 
 234             // the new sheet will be deleted. If it is not vetoed, the 
 235             // old sheet will be deleted and replaced with the new one. 
 236             buffer
->SetStyleSheetAndNotify(sheet
); 
 243         wxXmlNode
* child 
= node
->GetChildren(); 
 246             ImportXML(buffer
, child
); 
 247             child 
= child
->GetNext(); 
 254 bool wxRichTextXMLHandler::ImportStyleDefinition(wxRichTextStyleSheet
* sheet
, wxXmlNode
* node
) 
 256     wxString styleType 
= node
->GetName(); 
 257     wxString styleName 
= node
->GetPropVal(wxT("name"), wxEmptyString
); 
 258     wxString baseStyleName 
= node
->GetPropVal(wxT("basestyle"), wxEmptyString
); 
 260     if (styleName
.IsEmpty()) 
 263     if (styleType 
== wxT("characterstyle")) 
 265         wxRichTextCharacterStyleDefinition
* def 
= new wxRichTextCharacterStyleDefinition(styleName
); 
 266         def
->SetBaseStyle(baseStyleName
); 
 268         wxXmlNode
* child 
= node
->GetChildren(); 
 271             if (child
->GetName() == wxT("style")) 
 274                 GetStyle(attr
, child
, false); 
 277             child 
= child
->GetNext(); 
 280         sheet
->AddCharacterStyle(def
); 
 282     else if (styleType 
== wxT("paragraphstyle")) 
 284         wxRichTextParagraphStyleDefinition
* def 
= new wxRichTextParagraphStyleDefinition(styleName
); 
 286         wxString nextStyleName 
= node
->GetPropVal(wxT("nextstyle"), wxEmptyString
); 
 287         def
->SetNextStyle(nextStyleName
); 
 288         def
->SetBaseStyle(baseStyleName
); 
 290         wxXmlNode
* child 
= node
->GetChildren(); 
 293             if (child
->GetName() == wxT("style")) 
 296                 GetStyle(attr
, child
, false); 
 299             child 
= child
->GetNext(); 
 302         sheet
->AddParagraphStyle(def
); 
 304     else if (styleType 
== wxT("liststyle")) 
 306         wxRichTextListStyleDefinition
* def 
= new wxRichTextListStyleDefinition(styleName
); 
 308         wxString nextStyleName 
= node
->GetPropVal(wxT("nextstyle"), wxEmptyString
); 
 309         def
->SetNextStyle(nextStyleName
); 
 310         def
->SetBaseStyle(baseStyleName
); 
 312         wxXmlNode
* child 
= node
->GetChildren(); 
 315             if (child
->GetName() == wxT("style")) 
 318                 GetStyle(attr
, child
, false); 
 320                 wxString styleLevel 
= child
->GetPropVal(wxT("level"), wxEmptyString
); 
 321                 if (styleLevel
.IsEmpty()) 
 327                     int level 
= wxAtoi(styleLevel
); 
 328                     if (level 
> 0 && level 
<= 10) 
 330                         def
->SetLevelAttributes(level
-1, attr
); 
 334             child 
= child
->GetNext(); 
 337         sheet
->AddListStyle(def
); 
 343 //----------------------------------------------------------------------------- 
 344 //  xml support routines 
 345 //----------------------------------------------------------------------------- 
 347 bool wxRichTextXMLHandler::HasParam(wxXmlNode
* node
, const wxString
& param
) 
 349     return (GetParamNode(node
, param
) != NULL
); 
 352 wxXmlNode 
*wxRichTextXMLHandler::GetParamNode(wxXmlNode
* node
, const wxString
& param
) 
 354     wxCHECK_MSG(node
, NULL
, wxT("You can't access node data before it was initialized!")); 
 356     wxXmlNode 
*n 
= node
->GetChildren(); 
 360         if (n
->GetType() == wxXML_ELEMENT_NODE 
&& n
->GetName() == param
) 
 368 wxString 
wxRichTextXMLHandler::GetNodeContent(wxXmlNode 
*node
) 
 371     if (n 
== NULL
) return wxEmptyString
; 
 372     n 
= n
->GetChildren(); 
 376         if (n
->GetType() == wxXML_TEXT_NODE 
|| 
 377             n
->GetType() == wxXML_CDATA_SECTION_NODE
) 
 378             return n
->GetContent(); 
 381     return wxEmptyString
; 
 385 wxString 
wxRichTextXMLHandler::GetParamValue(wxXmlNode 
*node
, const wxString
& param
) 
 388         return GetNodeContent(node
); 
 390         return GetNodeContent(GetParamNode(node
, param
)); 
 393 wxString 
wxRichTextXMLHandler::GetText(wxXmlNode 
*node
, const wxString
& param
, bool WXUNUSED(translate
)) 
 395     wxXmlNode 
*parNode 
= GetParamNode(node
, param
); 
 398     wxString 
str1(GetNodeContent(parNode
)); 
 402 // For use with earlier versions of wxWidgets 
 403 #ifndef WXUNUSED_IN_UNICODE 
 405 #define WXUNUSED_IN_UNICODE(x) WXUNUSED(x) 
 407 #define WXUNUSED_IN_UNICODE(x) x 
 411 // write string to output: 
 412 inline static void OutputString(wxOutputStream
& stream
, const wxString
& str
, 
 413                                 wxMBConv 
*WXUNUSED_IN_UNICODE(convMem
) = NULL
, wxMBConv 
*convFile 
= NULL
) 
 415     if (str
.empty()) return; 
 419         const wxWX2MBbuf 
buf(str
.mb_str(*convFile
)); 
 420         stream
.Write((const char*)buf
, strlen((const char*)buf
)); 
 424         const wxWX2MBbuf 
buf(str
.mb_str(wxConvUTF8
)); 
 425         stream
.Write((const char*)buf
, strlen((const char*)buf
)); 
 428     if ( convFile 
== NULL 
) 
 429         stream
.Write(str
.mb_str(), str
.Len()); 
 432         wxString 
str2(str
.wc_str(*convMem
), *convFile
); 
 433         stream
.Write(str2
.mb_str(), str2
.Len()); 
 438 // Same as above, but create entities first. 
 439 // Translates '<' to "<", '>' to ">" and '&' to "&" 
 440 static void OutputStringEnt(wxOutputStream
& stream
, const wxString
& str
, 
 441                             wxMBConv 
*convMem 
= NULL
, wxMBConv 
*convFile 
= NULL
) 
 449     for (i 
= 0; i 
< len
; i
++) 
 453         // Original code excluded "&" but we _do_ want to convert 
 454         // the ampersand beginning & because otherwise when read in, 
 455         // the original "&" becomes "&". 
 457         if (c 
== wxT('<') || c 
== wxT('>') || c 
== wxT('"') || 
 458             (c 
== wxT('&') /* && (str.Mid(i+1, 4) != wxT("amp;")) */ )) 
 460             OutputString(stream
, str
.Mid(last
, i 
- last
), convMem
, convFile
); 
 464                 OutputString(stream
, wxT("<"), NULL
, NULL
); 
 467                 OutputString(stream
, wxT(">"), NULL
, NULL
); 
 470                 OutputString(stream
, wxT("&"), NULL
, NULL
); 
 473                 OutputString(stream
, wxT("""), NULL
, NULL
); 
 479         else if (wxUChar(c
) > 127) 
 481             OutputString(stream
, str
.Mid(last
, i 
- last
), convMem
, convFile
); 
 483             wxString 
s(wxT("&#")); 
 486             OutputString(stream
, s
, NULL
, NULL
); 
 490     OutputString(stream
, str
.Mid(last
, i 
- last
), convMem
, convFile
); 
 493 inline static void OutputIndentation(wxOutputStream
& stream
, int indent
) 
 495     wxString str 
= wxT("\n"); 
 496     for (int i 
= 0; i 
< indent
; i
++) 
 497         str 
<< wxT(' ') << wxT(' '); 
 498     OutputString(stream
, str
, NULL
, NULL
); 
 501 // Convert a colour to a 6-digit hex string 
 502 static wxString 
ColourToHexString(const wxColour
& col
) 
 506     hex 
+= wxDecToHex(col
.Red()); 
 507     hex 
+= wxDecToHex(col
.Green()); 
 508     hex 
+= wxDecToHex(col
.Blue()); 
 513 // Convert 6-digit hex string to a colour 
 514 static wxColour 
HexStringToColour(const wxString
& hex
) 
 516     unsigned char r 
= (unsigned char)wxHexToDec(hex
.Mid(0, 2)); 
 517     unsigned char g 
= (unsigned char)wxHexToDec(hex
.Mid(2, 2)); 
 518     unsigned char b 
= (unsigned char)wxHexToDec(hex
.Mid(4, 2)); 
 520     return wxColour(r
, g
, b
); 
 523 bool wxRichTextXMLHandler::DoSaveFile(wxRichTextBuffer 
*buffer
, wxOutputStream
& stream
) 
 528     wxString 
version(wxT("1.0") ) ; 
 530     bool deleteConvFile 
= false; 
 531     wxString fileEncoding
; 
 532     wxMBConv
* convFile 
= NULL
; 
 535     fileEncoding 
= wxT("UTF-8"); 
 536     convFile 
= & wxConvUTF8
; 
 538     fileEncoding 
= wxT("ISO-8859-1"); 
 539     convFile 
= & wxConvISO8859_1
; 
 542     // If SetEncoding has been called, change the output encoding. 
 543     if (!m_encoding
.empty() && m_encoding
.Lower() != fileEncoding
.Lower()) 
 545         if (m_encoding 
== wxT("<System>")) 
 547             fileEncoding 
= wxLocale::GetSystemEncodingName(); 
 551             fileEncoding 
= m_encoding
; 
 554         // GetSystemEncodingName may not have returned a name 
 555         if (fileEncoding
.empty()) 
 557             fileEncoding 
= wxT("UTF-8"); 
 559             fileEncoding 
= wxT("ISO-8859-1"); 
 561         convFile 
= new wxCSConv(fileEncoding
); 
 562         deleteConvFile 
= true; 
 566     wxMBConv
* convMem 
= wxConvCurrent
; 
 568     wxMBConv
* convMem 
= NULL
; 
 572     s
.Printf(wxT("<?xml version=\"%s\" encoding=\"%s\"?>\n"), 
 573         (const wxChar
*) version
, (const wxChar
*) fileEncoding 
); 
 574     OutputString(stream
, s
, NULL
, NULL
); 
 575     OutputString(stream
, wxT("<richtext version=\"1.0.0.0\" xmlns=\"http://www.wxwidgets.org\">") , NULL
, NULL
); 
 579     if (buffer
->GetStyleSheet() && (GetFlags() & wxRICHTEXT_HANDLER_INCLUDE_STYLESHEET
)) 
 581         OutputIndentation(stream
, level
); 
 582         wxString nameAndDescr
; 
 583         if (!buffer
->GetStyleSheet()->GetName().IsEmpty()) 
 584             nameAndDescr 
<< wxT(" name=\"") << buffer
->GetStyleSheet()->GetName() << wxT("\""); 
 585         if (!buffer
->GetStyleSheet()->GetDescription().IsEmpty()) 
 586             nameAndDescr 
<< wxT(" description=\"") << buffer
->GetStyleSheet()->GetDescription() << wxT("\""); 
 587         OutputString(stream
, wxString(wxT("<stylesheet")) + nameAndDescr 
+ wxT(">"), convMem
, convFile
); 
 591         for (i 
= 0; i 
< (int) buffer
->GetStyleSheet()->GetCharacterStyleCount(); i
++) 
 593             wxRichTextCharacterStyleDefinition
* def 
= buffer
->GetStyleSheet()->GetCharacterStyle(i
); 
 594             ExportStyleDefinition(stream
, convMem
, convFile
, def
, level 
+ 1); 
 597         for (i 
= 0; i 
< (int) buffer
->GetStyleSheet()->GetParagraphStyleCount(); i
++) 
 599             wxRichTextParagraphStyleDefinition
* def 
= buffer
->GetStyleSheet()->GetParagraphStyle(i
); 
 600             ExportStyleDefinition(stream
, convMem
, convFile
, def
, level 
+ 1); 
 603         for (i 
= 0; i 
< (int) buffer
->GetStyleSheet()->GetListStyleCount(); i
++) 
 605             wxRichTextListStyleDefinition
* def 
= buffer
->GetStyleSheet()->GetListStyle(i
); 
 606             ExportStyleDefinition(stream
, convMem
, convFile
, def
, level 
+ 1); 
 609         OutputIndentation(stream
, level
); 
 610         OutputString(stream
, wxT("</stylesheet>"), convMem
, convFile
); 
 614     bool success 
= ExportXML(stream
, convMem
, convFile
, *buffer
, level
); 
 616     OutputString(stream
, wxT("\n</richtext>") , NULL
, NULL
); 
 617     OutputString(stream
, wxT("\n"), NULL
, NULL
); 
 625 /// Recursively export an object 
 626 bool wxRichTextXMLHandler::ExportXML(wxOutputStream
& stream
, wxMBConv
* convMem
, wxMBConv
* convFile
, wxRichTextObject
& obj
, int indent
) 
 629     if (obj
.IsKindOf(CLASSINFO(wxRichTextParagraphLayoutBox
))) 
 630         objectName 
= wxT("paragraphlayout"); 
 631     else if (obj
.IsKindOf(CLASSINFO(wxRichTextParagraph
))) 
 632         objectName 
= wxT("paragraph"); 
 633     else if (obj
.IsKindOf(CLASSINFO(wxRichTextPlainText
))) 
 634         objectName 
= wxT("text"); 
 635     else if (obj
.IsKindOf(CLASSINFO(wxRichTextImage
))) 
 636         objectName 
= wxT("image"); 
 638         objectName 
= wxT("object"); 
 640     bool terminateTag 
= true; 
 642     if (obj
.IsKindOf(CLASSINFO(wxRichTextPlainText
))) 
 644         wxRichTextPlainText
& textObj 
= (wxRichTextPlainText
&) obj
; 
 646         wxString style 
= CreateStyle(obj
.GetAttributes(), false); 
 650         const wxString
& text 
= textObj
.GetText(); 
 651         int len 
= (int) text
.Length(); 
 652         for (i 
= 0; i 
< len
; i
++) 
 654             int c 
= (int) text
[i
]; 
 655             if (c 
< 32 && c 
!= 9 && c 
!= 10 && c 
!= 13) 
 659                     OutputIndentation(stream
, indent
); 
 660                     OutputString(stream
, wxT("<") + objectName
, convMem
, convFile
); 
 662                     OutputString(stream
, style 
+ wxT(">"), convMem
, convFile
); 
 664                     wxString 
fragment(text
.Mid(last
, i
-last
)); 
 665                     if (!fragment
.empty() && (fragment
[0] == wxT(' ') || fragment
[fragment
.length()-1] == wxT(' '))) 
 667                         OutputString(stream
, wxT("\""), convMem
, convFile
); 
 668                         OutputStringEnt(stream
, fragment
, convMem
, convFile
); 
 669                         OutputString(stream
, wxT("\""), convMem
, convFile
); 
 672                         OutputStringEnt(stream
, fragment
, convMem
, convFile
); 
 674                     OutputString(stream
, wxT("</text>"), convMem
, convFile
); 
 678                 // Output this character as a number in a separate tag, because XML can't cope 
 679                 // with entities below 32 except for 9, 10 and 13 
 681                 OutputIndentation(stream
, indent
); 
 682                 OutputString(stream
, wxT("<symbol"), convMem
, convFile
); 
 684                 OutputString(stream
, style 
+ wxT(">"), convMem
, convFile
); 
 685                 OutputString(stream
, wxString::Format(wxT("%d"), c
), convMem
, convFile
); 
 687                 OutputString(stream
, wxT("</symbol>"), convMem
, convFile
); 
 695             fragment 
= text
.Mid(last
, i
-last
); 
 699             OutputIndentation(stream
, indent
); 
 700             OutputString(stream
, wxT("<") + objectName
, convMem
, convFile
); 
 702             OutputString(stream
, style 
+ wxT(">"), convMem
, convFile
); 
 704             if (!fragment
.empty() && (fragment
[0] == wxT(' ') || fragment
[fragment
.length()-1] == wxT(' '))) 
 706                 OutputString(stream
, wxT("\""), convMem
, convFile
); 
 707                 OutputStringEnt(stream
, fragment
, convMem
, convFile
); 
 708                 OutputString(stream
, wxT("\""), convMem
, convFile
); 
 711                 OutputStringEnt(stream
, fragment
, convMem
, convFile
); 
 714             terminateTag 
= false; 
 716     else if (obj
.IsKindOf(CLASSINFO(wxRichTextImage
))) 
 718         wxRichTextImage
& imageObj 
= (wxRichTextImage
&) obj
; 
 720         if (imageObj
.GetImage().Ok() && !imageObj
.GetImageBlock().Ok()) 
 721             imageObj
.MakeBlock(); 
 723         OutputIndentation(stream
, indent
); 
 724         OutputString(stream
, wxT("<") + objectName
, convMem
, convFile
); 
 725         if (!imageObj
.GetImageBlock().Ok()) 
 728             OutputString(stream
, wxT(">"), convMem
, convFile
); 
 732             OutputString(stream
, wxString::Format(wxT(" imagetype=\"%d\">"), (int) imageObj
.GetImageBlock().GetImageType())); 
 735         OutputIndentation(stream
, indent
+1); 
 736         OutputString(stream
, wxT("<data>"), convMem
, convFile
); 
 738         imageObj
.GetImageBlock().WriteHex(stream
); 
 740         OutputString(stream
, wxT("</data>"), convMem
, convFile
); 
 742     else if (obj
.IsKindOf(CLASSINFO(wxRichTextCompositeObject
))) 
 744         OutputIndentation(stream
, indent
); 
 745         OutputString(stream
, wxT("<") + objectName
, convMem
, convFile
); 
 748         if (objectName 
== wxT("paragraph") || objectName 
== wxT("paragraphlayout")) 
 751         wxString style 
= CreateStyle(obj
.GetAttributes(), isPara
); 
 753         if (objectName 
== wxT("paragraphlayout") && ((wxRichTextParagraphLayoutBox
&) obj
).GetPartialParagraph()) 
 754             style 
<< wxT(" partialparagraph=\"true\""); 
 756         OutputString(stream
, style 
+ wxT(">"), convMem
, convFile
); 
 758         wxRichTextCompositeObject
& composite 
= (wxRichTextCompositeObject
&) obj
; 
 760         for (i 
= 0; i 
< composite
.GetChildCount(); i
++) 
 762             wxRichTextObject
* child 
= composite
.GetChild(i
); 
 763             ExportXML(stream
, convMem
, convFile
, *child
, indent
+1); 
 767     if (objectName 
!= wxT("text")) 
 768         OutputIndentation(stream
, indent
); 
 771         OutputString(stream
, wxT("</") + objectName 
+ wxT(">"), convMem
, convFile
); 
 776 bool wxRichTextXMLHandler::ExportStyleDefinition(wxOutputStream
& stream
, wxMBConv
* convMem
, wxMBConv
* convFile
, wxRichTextStyleDefinition
* def
, int level
) 
 778     wxRichTextCharacterStyleDefinition
* charDef 
= wxDynamicCast(def
, wxRichTextCharacterStyleDefinition
); 
 779     wxRichTextParagraphStyleDefinition
* paraDef 
= wxDynamicCast(def
, wxRichTextParagraphStyleDefinition
); 
 780     wxRichTextListStyleDefinition
* listDef 
= wxDynamicCast(def
, wxRichTextListStyleDefinition
); 
 782     wxString baseStyle 
= def
->GetBaseStyle(); 
 783     wxString baseStyleProp
; 
 784     if (!baseStyle
.IsEmpty()) 
 785         baseStyleProp 
= wxT(" basestyle=\"") + baseStyle 
+ wxT("\""); 
 787     wxString descr 
= def
->GetDescription(); 
 789     if (!descr
.IsEmpty()) 
 790         descrProp 
= wxT(" description=\"") + descr 
+ wxT("\""); 
 794         OutputIndentation(stream
, level
); 
 795         OutputString(stream
, wxT("<characterstyle") + baseStyleProp 
+ descrProp 
+ wxT(">"), convMem
, convFile
); 
 799         wxString style 
= CreateStyle(def
->GetStyle(), false); 
 801         OutputIndentation(stream
, level
); 
 802         OutputString(stream
, wxT("<style ") + style 
+ wxT(">"), convMem
, convFile
); 
 804         OutputIndentation(stream
, level
); 
 805         OutputString(stream
, wxT("</style>"), convMem
, convFile
); 
 809         OutputIndentation(stream
, level
); 
 810         OutputString(stream
, wxT("</characterstyle>"), convMem
, convFile
); 
 814         OutputIndentation(stream
, level
); 
 816         if (!listDef
->GetNextStyle().IsEmpty()) 
 817             baseStyleProp 
<< wxT(" basestyle=\"") << listDef
->GetNextStyle() << wxT("\""); 
 819         OutputString(stream
, wxT("<liststyle") + baseStyleProp 
+ descrProp 
+ wxT(">"), convMem
, convFile
); 
 823         wxString style 
= CreateStyle(def
->GetStyle(), false); 
 825         OutputIndentation(stream
, level
); 
 826         OutputString(stream
, wxT("<style ") + style 
+ wxT(">"), convMem
, convFile
); 
 828         OutputIndentation(stream
, level
); 
 829         OutputString(stream
, wxT("</style>"), convMem
, convFile
); 
 832         for (i 
= 0; i 
< 10; i 
++) 
 834             wxRichTextAttr
* levelAttr 
= listDef
->GetLevelAttributes(i
); 
 837                 wxString style 
= CreateStyle(def
->GetStyle(), false); 
 838                 wxString levelStr 
= wxString::Format(wxT(" level=\"%d\" "), (i
+1)); 
 840                 OutputIndentation(stream
, level
); 
 841                 OutputString(stream
, wxT("<style ") + levelStr 
+ style 
+ wxT(">"), convMem
, convFile
); 
 843                 OutputIndentation(stream
, level
); 
 844                 OutputString(stream
, wxT("</style>"), convMem
, convFile
); 
 850         OutputIndentation(stream
, level
); 
 851         OutputString(stream
, wxT("</liststyle>"), convMem
, convFile
); 
 855         OutputIndentation(stream
, level
); 
 857         if (!paraDef
->GetNextStyle().IsEmpty()) 
 858             baseStyleProp 
<< wxT(" basestyle=\"") << paraDef
->GetNextStyle() << wxT("\""); 
 860         OutputString(stream
, wxT("<paragraphstyle") + baseStyleProp 
+ descrProp 
+ wxT(">"), convMem
, convFile
); 
 864         wxString style 
= CreateStyle(def
->GetStyle(), false); 
 866         OutputIndentation(stream
, level
); 
 867         OutputString(stream
, wxT("<style ") + style 
+ wxT(">"), convMem
, convFile
); 
 869         OutputIndentation(stream
, level
); 
 870         OutputString(stream
, wxT("</style>"), convMem
, convFile
); 
 874         OutputIndentation(stream
, level
); 
 875         OutputString(stream
, wxT("</paragraphstyle>"), convMem
, convFile
); 
 881 /// Create style parameters 
 882 wxString 
wxRichTextXMLHandler::CreateStyle(const wxTextAttrEx
& attr
, bool isPara
) 
 885     if (attr
.HasTextColour() && attr
.GetTextColour().Ok()) 
 887         str 
<< wxT(" textcolor=\"#") << ColourToHexString(attr
.GetTextColour()) << wxT("\""); 
 889     if (attr
.HasBackgroundColour() && attr
.GetBackgroundColour().Ok()) 
 891         str 
<< wxT(" bgcolor=\"#") << ColourToHexString(attr
.GetBackgroundColour()) << wxT("\""); 
 894     if (attr
.GetFont().Ok()) 
 896         if (attr
.HasFontSize()) 
 897             str 
<< wxT(" fontsize=\"") << attr
.GetFont().GetPointSize() << wxT("\""); 
 899         //if (attr.HasFontFamily()) 
 900         //    str << wxT(" fontfamily=\"") << attr.GetFont().GetFamily() << wxT("\""); 
 902         if (attr
.HasFontItalic()) 
 903             str 
<< wxT(" fontstyle=\"") << attr
.GetFont().GetStyle() << wxT("\""); 
 905         if (attr
.HasFontWeight()) 
 906             str 
<< wxT(" fontweight=\"") << attr
.GetFont().GetWeight() << wxT("\""); 
 908         if (attr
.HasFontUnderlined()) 
 909             str 
<< wxT(" fontunderlined=\"") << (int) attr
.GetFont().GetUnderlined() << wxT("\""); 
 911         if (attr
.HasFontFaceName()) 
 912             str 
<< wxT(" fontface=\"") << attr
.GetFont().GetFaceName() << wxT("\""); 
 915     if (attr
.HasTextEffects()) 
 917         str 
<< wxT(" texteffects=\""); 
 918         str 
<< attr
.GetTextEffects(); 
 921         str 
<< wxT(" texteffectflags=\""); 
 922         str 
<< attr
.GetTextEffectFlags(); 
 926     if (!attr
.GetCharacterStyleName().empty()) 
 927         str 
<< wxT(" characterstyle=\"") << wxString(attr
.GetCharacterStyleName()) << wxT("\""); 
 931         if (attr
.HasAlignment()) 
 932             str 
<< wxT(" alignment=\"") << (int) attr
.GetAlignment() << wxT("\""); 
 934         if (attr
.HasLeftIndent()) 
 936             str 
<< wxT(" leftindent=\"") << (int) attr
.GetLeftIndent() << wxT("\""); 
 937             str 
<< wxT(" leftsubindent=\"") << (int) attr
.GetLeftSubIndent() << wxT("\""); 
 940         if (attr
.HasRightIndent()) 
 941             str 
<< wxT(" rightindent=\"") << (int) attr
.GetRightIndent() << wxT("\""); 
 943         if (attr
.HasParagraphSpacingAfter()) 
 944             str 
<< wxT(" parspacingafter=\"") << (int) attr
.GetParagraphSpacingAfter() << wxT("\""); 
 946         if (attr
.HasParagraphSpacingBefore()) 
 947             str 
<< wxT(" parspacingbefore=\"") << (int) attr
.GetParagraphSpacingBefore() << wxT("\""); 
 949         if (attr
.HasLineSpacing()) 
 950             str 
<< wxT(" linespacing=\"") << (int) attr
.GetLineSpacing() << wxT("\""); 
 952         if (attr
.HasBulletStyle()) 
 953             str 
<< wxT(" bulletstyle=\"") << (int) attr
.GetBulletStyle() << wxT("\""); 
 955         if (attr
.HasBulletNumber()) 
 956             str 
<< wxT(" bulletnumber=\"") << (int) attr
.GetBulletNumber() << wxT("\""); 
 958         if (attr
.HasBulletText()) 
 960             // If using a bullet symbol, convert to integer in case it's a non-XML-friendly character. 
 961             // Otherwise, assume it's XML-friendly text such as outline numbering, e.g. 1.2.3.1 
 962             if (!attr
.GetBulletText().IsEmpty() && (attr
.GetBulletStyle() & wxTEXT_ATTR_BULLET_STYLE_SYMBOL
)) 
 963                 str 
<< wxT(" bulletsymbol=\"") << (int) (attr
.GetBulletText()[0]) << wxT("\""); 
 965                 str 
<< wxT(" bullettext=\"") << attr
.GetBulletText() << wxT("\""); 
 967             str 
<< wxT(" bulletfont=\"") << attr
.GetBulletFont() << wxT("\""); 
 970         if (attr
.HasBulletName()) 
 971             str 
<< wxT(" bulletname=\"") << attr
.GetBulletName() << wxT("\""); 
 974             str 
<< wxT(" url=\"") << attr
.GetURL() << wxT("\""); 
 976         if (!attr
.GetParagraphStyleName().empty()) 
 977             str 
<< wxT(" parstyle=\"") << wxString(attr
.GetParagraphStyleName()) << wxT("\""); 
 979         if (!attr
.GetListStyleName().empty()) 
 980             str 
<< wxT(" liststyle=\"") << wxString(attr
.GetListStyleName()) << wxT("\""); 
 984             str 
<< wxT(" tabs=\""); 
 986             for (i 
= 0; i 
< attr
.GetTabs().GetCount(); i
++) 
 990                 str 
<< attr
.GetTabs()[i
]; 
 995         if (attr
.HasPageBreak()) 
 997             str 
<< wxT(" pagebreak=\"1\""); 
1000         if (attr
.HasOutlineLevel()) 
1001             str 
<< wxT(" outlinelevel=\"") << (int) attr
.GetOutlineLevel() << wxT("\""); 
1008 /// Get style parameters 
1009 bool wxRichTextXMLHandler::GetStyle(wxTextAttrEx
& attr
, wxXmlNode
* node
, bool isPara
) 
1011     wxString fontFacename
; 
1013     int fontFamily 
= wxDEFAULT
; 
1014     int fontWeight 
= wxNORMAL
; 
1015     int fontStyle 
= wxNORMAL
; 
1016     bool fontUnderlined 
= false; 
1020     fontFacename 
= node
->GetPropVal(wxT("fontface"), wxEmptyString
); 
1021     if (!fontFacename
.IsEmpty()) 
1022         fontFlags 
|= wxTEXT_ATTR_FONT_FACE
; 
1025     //value = node->GetPropVal(wxT("fontfamily"), wxEmptyString); 
1026     //if (!value.empty()) 
1027     //    fontFamily = wxAtoi(value); 
1029     value 
= node
->GetPropVal(wxT("fontstyle"), wxEmptyString
); 
1032         fontStyle 
= wxAtoi(value
); 
1033         fontFlags 
|= wxTEXT_ATTR_FONT_ITALIC
; 
1036     value 
= node
->GetPropVal(wxT("fontsize"), wxEmptyString
); 
1039         fontSize 
= wxAtoi(value
); 
1040         fontFlags 
|= wxTEXT_ATTR_FONT_SIZE
; 
1043     value 
= node
->GetPropVal(wxT("fontweight"), wxEmptyString
); 
1046         fontWeight 
= wxAtoi(value
); 
1047         fontFlags 
|= wxTEXT_ATTR_FONT_WEIGHT
; 
1050     value 
= node
->GetPropVal(wxT("fontunderlined"), wxEmptyString
); 
1053         fontUnderlined 
= wxAtoi(value
) != 0; 
1054         fontFlags 
|= wxTEXT_ATTR_FONT_UNDERLINE
; 
1057     attr
.SetFlags(fontFlags
); 
1059     if (attr
.HasFlag(wxTEXT_ATTR_FONT
)) 
1060         attr
.SetFont(* wxTheFontList
->FindOrCreateFont(fontSize
, fontFamily
, fontStyle
, fontWeight
, fontUnderlined
, fontFacename
)); 
1062     // Restore correct font flags 
1063     attr
.SetFlags(fontFlags
); 
1065     value 
= node
->GetPropVal(wxT("textcolor"), wxEmptyString
); 
1068         if (value
[0] == wxT('#')) 
1069             attr
.SetTextColour(HexStringToColour(value
.Mid(1))); 
1071             attr
.SetTextColour(value
); 
1074     value 
= node
->GetPropVal(wxT("backgroundcolor"), wxEmptyString
); 
1077         if (value
[0] == wxT('#')) 
1078             attr
.SetBackgroundColour(HexStringToColour(value
.Mid(1))); 
1080             attr
.SetBackgroundColour(value
); 
1083     value 
= node
->GetPropVal(wxT("characterstyle"), wxEmptyString
); 
1085         attr
.SetCharacterStyleName(value
); 
1087     value 
= node
->GetPropVal(wxT("texteffects"), wxEmptyString
); 
1088     if (!value
.IsEmpty()) 
1090         attr
.SetTextEffects(wxAtoi(value
)); 
1093     value 
= node
->GetPropVal(wxT("texteffectflags"), wxEmptyString
); 
1094     if (!value
.IsEmpty()) 
1096         attr
.SetTextEffectFlags(wxAtoi(value
)); 
1099     // Set paragraph attributes 
1102         value 
= node
->GetPropVal(wxT("alignment"), wxEmptyString
); 
1104             attr
.SetAlignment((wxTextAttrAlignment
) wxAtoi(value
)); 
1106         int leftSubIndent 
= 0; 
1108         bool hasLeftIndent 
= false; 
1110         value 
= node
->GetPropVal(wxT("leftindent"), wxEmptyString
); 
1113             leftIndent 
= wxAtoi(value
); 
1114             hasLeftIndent 
= true; 
1117         value 
= node
->GetPropVal(wxT("leftsubindent"), wxEmptyString
); 
1120             leftSubIndent 
= wxAtoi(value
); 
1121             hasLeftIndent 
= true; 
1125             attr
.SetLeftIndent(leftIndent
, leftSubIndent
); 
1127         value 
= node
->GetPropVal(wxT("rightindent"), wxEmptyString
); 
1129             attr
.SetRightIndent(wxAtoi(value
)); 
1131         value 
= node
->GetPropVal(wxT("parspacingbefore"), wxEmptyString
); 
1133             attr
.SetParagraphSpacingBefore(wxAtoi(value
)); 
1135         value 
= node
->GetPropVal(wxT("parspacingafter"), wxEmptyString
); 
1137             attr
.SetParagraphSpacingAfter(wxAtoi(value
)); 
1139         value 
= node
->GetPropVal(wxT("linespacing"), wxEmptyString
); 
1141             attr
.SetLineSpacing(wxAtoi(value
)); 
1143         value 
= node
->GetPropVal(wxT("bulletstyle"), wxEmptyString
); 
1145             attr
.SetBulletStyle(wxAtoi(value
)); 
1147         value 
= node
->GetPropVal(wxT("bulletnumber"), wxEmptyString
); 
1149             attr
.SetBulletNumber(wxAtoi(value
)); 
1151         value 
= node
->GetPropVal(wxT("bulletsymbol"), wxEmptyString
); 
1154             wxChar ch 
= wxAtoi(value
); 
1157             attr
.SetBulletText(s
); 
1160         value 
= node
->GetPropVal(wxT("bullettext"), wxEmptyString
); 
1162             attr
.SetBulletText(value
); 
1164         value 
= node
->GetPropVal(wxT("bulletfont"), wxEmptyString
); 
1166             attr
.SetBulletFont(value
); 
1168         value 
= node
->GetPropVal(wxT("bulletname"), wxEmptyString
); 
1170             attr
.SetBulletName(value
); 
1172         value 
= node
->GetPropVal(wxT("url"), wxEmptyString
); 
1176         value 
= node
->GetPropVal(wxT("parstyle"), wxEmptyString
); 
1178             attr
.SetParagraphStyleName(value
); 
1180         value 
= node
->GetPropVal(wxT("liststyle"), wxEmptyString
); 
1182             attr
.SetListStyleName(value
); 
1184         value 
= node
->GetPropVal(wxT("tabs"), wxEmptyString
); 
1188             wxStringTokenizer 
tkz(value
, wxT(",")); 
1189             while (tkz
.HasMoreTokens()) 
1191                 wxString token 
= tkz
.GetNextToken(); 
1192                 tabs
.Add(wxAtoi(token
)); 
1197         value 
= node
->GetPropVal(wxT("pagebreak"), wxEmptyString
); 
1198         if (!value
.IsEmpty()) 
1200             attr
.SetPageBreak(wxAtoi(value
) != 0); 
1203         value 
= node
->GetPropVal(wxT("outlinelevel"), wxEmptyString
); 
1204         if (!value
.IsEmpty()) 
1206             attr
.SetOutlineLevel(wxAtoi(value
) != 0); 
1217     // wxUSE_RICHTEXT && wxUSE_XML