diff --git a/docs/latex/wx/richtexthtmlhandler.tex b/docs/latex/wx/richtexthtmlhandler.tex index fb0cd23d29..f0808f8a72 100644 --- a/docs/latex/wx/richtexthtmlhandler.tex +++ b/docs/latex/wx/richtexthtmlhandler.tex @@ -74,6 +74,12 @@ for example after the user has viewed the HTML file. Saves the buffer content to the HTML stream. +\membersection{wxRichTextHTMLHandler::GetFontSizeMapping}\label{wxrichtexthtmlhandlergetfontsizemapping} + +\func{wxArrayInt}{GetFontSizeMapping}{\void} + +Returns the mapping for converting point sizes to HTML font sizes. + \membersection{wxRichTextHTMLHandler::GetTempDir}\label{wxrichtexthtmlhandlergettempdir} \constfunc{const wxString\&}{GetTempDir}{\void} @@ -92,6 +98,29 @@ Returns the image locations for the last operation. Reset the file counter, in case, for example, the same names are required each time +\membersection{wxRichTextHTMLHandler::SetFontSizeMapping}\label{wxrichtexthtmlhandlersetfontsizemapping} + +\func{void}{SetFontSizeMapping}{\param{const wxArrayInt\& }{fontSizeMapping}} + +Sets the mapping for converting point sizes to HTML font sizes. +There should be 7 elements, one for each HTML font size, each element specifying the maximum point size for that +HTML font size. + +For example: + +\begin{verbatim} + wxArrayInt fontSizeMapping; + fontSizeMapping.Add(7); + fontSizeMapping.Add(9); + fontSizeMapping.Add(11); + fontSizeMapping.Add(12); + fontSizeMapping.Add(14); + fontSizeMapping.Add(22); + fontSizeMapping.Add(100); + + htmlHandler.SetFontSizeMapping(fontSizeMapping); +\end{verbatim} + \membersection{wxRichTextHTMLHandler::SetTempDir}\label{wxrichtexthtmlhandlersettempdir} \func{void}{SetTempDir}{\param{const wxString\& }{tempDir}} diff --git a/include/wx/richtext/richtexthtml.h b/include/wx/richtext/richtexthtml.h index 20c86ed60a..f71ed68d9d 100644 --- a/include/wx/richtext/richtexthtml.h +++ b/include/wx/richtext/richtexthtml.h @@ -26,9 +26,7 @@ class WXDLLIMPEXP_RICHTEXT wxRichTextHTMLHandler: public wxRichTextFileHandler { DECLARE_CLASS(wxRichTextHTMLHandler) public: - wxRichTextHTMLHandler(const wxString& name = wxT("HTML"), const wxString& ext = wxT("html"), int type = wxRICHTEXT_TYPE_HTML) - : wxRichTextFileHandler(name, ext, type), m_indent(0), m_font(false), m_list(false), m_is_ul(false) - { } + wxRichTextHTMLHandler(const wxString& name = wxT("HTML"), const wxString& ext = wxT("html"), int type = wxRICHTEXT_TYPE_HTML); /// Can we save using this handler? virtual bool CanSave() const { return true; } @@ -64,6 +62,12 @@ public: void SetTempDir(const wxString& tempDir) { m_tempDir = tempDir; } const wxString& GetTempDir() const { return m_tempDir; } + /// Set and get mapping from point size to HTML font size. There should be 7 elements, + /// one for each HTML font size, each element specifying the maximum point size for that + /// HTML font size. E.g. 8, 10, 13, 17, 22, 29, 100 + void SetFontSizeMapping(const wxArrayInt& fontSizeMapping) { m_fontSizeMapping = fontSizeMapping; } + wxArrayInt GetFontSizeMapping() const { return m_fontSizeMapping; } + protected: // Implementation @@ -73,21 +77,22 @@ protected: virtual bool DoSaveFile(wxRichTextBuffer *buffer, wxOutputStream& stream); /// Output character formatting - virtual void BeginCharacterFormatting(const wxTextAttrEx& currentStyle, const wxTextAttrEx& thisStyle, const wxTextAttrEx& paraStyle, wxOutputStream& stream ); - virtual void EndCharacterFormatting(const wxTextAttrEx& WXUNUSED(currentStyle), const wxTextAttrEx& thisStyle, const wxTextAttrEx& paraStyle, wxOutputStream& stream ); + void BeginCharacterFormatting(const wxTextAttrEx& currentStyle, const wxTextAttrEx& thisStyle, const wxTextAttrEx& paraStyle, wxTextOutputStream& stream ); + void EndCharacterFormatting(const wxTextAttrEx& currentStyle, const wxTextAttrEx& thisStyle, const wxTextAttrEx& paraStyle, wxTextOutputStream& stream ); /// Output paragraph formatting - virtual void OutputParagraphFormatting(const wxTextAttrEx& WXUNUSED(currentStyle), const wxTextAttrEx& thisStyle, wxOutputStream& stream/*, bool start*/); + void BeginParagraphFormatting(const wxTextAttrEx& currentStyle, const wxTextAttrEx& thisStyle, wxTextOutputStream& stream); + void EndParagraphFormatting(const wxTextAttrEx& currentStyle, const wxTextAttrEx& thisStyle, wxTextOutputStream& stream); + + /// Output font tag + void OutputFont(const wxTextAttrEx& style, wxTextOutputStream& stream); + + /// Closes lists to level (-1 means close all) + void CloseLists(int level, wxTextOutputStream& str); /// Writes an image to its base64 equivalent, or to the memory filesystem, or to a file void WriteImage(wxRichTextImage* image, wxOutputStream& stream); - /// Builds required indentation - void Indent(const wxTextAttrEx& thisStyle, wxTextOutputStream& str); - - /// Left indent - void LIndent(const wxTextAttrEx& thisStyle, wxTextOutputStream& str); - /// Converts from pt to size property compatible height long PtToSize(long size); @@ -104,28 +109,24 @@ protected: wxString SymbolicIndent(long indent); /// Finds the html equivalent of the specified bullet - void TypeOfList(const wxTextAttrEx& thisStyle, wxString& tag); - - /// Closes existings or Opens new tables for navigation to an item's horizontal position. - void NavigateToListPosition(const wxTextAttrEx& thisStyle, wxTextOutputStream& str); + int TypeOfList(const wxTextAttrEx& thisStyle, wxString& tag); #endif // Data members + wxRichTextBuffer* m_buffer; + /// Indentation values of the table tags wxArrayInt m_indents; - /// Horizontal position of the current table - long m_indent; + /// Stack of list types: 0 = ol, 1 = ul + wxArrayInt m_listTypes; - /// Is there any opened font tag + /// Is there any opened font tag? bool m_font; - /// Is there any opened ul/ol tag - bool m_list; - - /// type of list, ul or ol? - bool m_is_ul; + /// Are we in a table? + bool m_inTable; /// A list of the image files or in-memory images created by the last operation. wxArrayString m_imageLocations; @@ -133,6 +134,9 @@ protected: /// A location for the temporary files wxString m_tempDir; + /// A mapping from point size to HTML font size + wxArrayInt m_fontSizeMapping; + /// A counter for generating filenames static int sm_fileCounter; }; diff --git a/samples/richtext/richtext.cpp b/samples/richtext/richtext.cpp index ec18b3373a..9577212f44 100644 --- a/samples/richtext/richtext.cpp +++ b/samples/richtext/richtext.cpp @@ -724,9 +724,10 @@ MyFrame::MyFrame(const wxString& title, wxWindowID id, const wxPoint& pos, r.BeginBold(); r.BeginFontSize(14); - r.WriteText(wxT("Welcome to wxRichTextCtrl, a wxWidgets control")); - r.LineBreak(); - r.WriteText(wxT("for editing and presenting styled text and images")); + + wxString lineBreak = (wxChar) 29; + + r.WriteText(wxString(wxT("Welcome to wxRichTextCtrl, a wxWidgets control")) + lineBreak + wxT("for editing and presenting styled text and images")); r.EndFontSize(); r.Newline(); @@ -1331,6 +1332,17 @@ void MyFrame::OnViewHTML(wxCommandEvent& WXUNUSED(event)) wxRichTextHTMLHandler htmlHandler; htmlHandler.SetFlags(wxRICHTEXT_HANDLER_SAVE_IMAGES_TO_MEMORY); + + wxArrayInt fontSizeMapping; + fontSizeMapping.Add(7); + fontSizeMapping.Add(9); + fontSizeMapping.Add(11); + fontSizeMapping.Add(12); + fontSizeMapping.Add(14); + fontSizeMapping.Add(22); + fontSizeMapping.Add(100); + + htmlHandler.SetFontSizeMapping(fontSizeMapping); if (htmlHandler.SaveFile(& m_richTextCtrl->GetBuffer(), strStream)) { diff --git a/src/richtext/richtexthtml.cpp b/src/richtext/richtexthtml.cpp index 95ed0c1d4c..a24b6f9db8 100644 --- a/src/richtext/richtexthtml.cpp +++ b/src/richtext/richtexthtml.cpp @@ -19,6 +19,7 @@ #if wxUSE_RICHTEXT #include "wx/richtext/richtexthtml.h" +#include "wx/richtext/richtextstyles.h" #ifndef WX_PRECOMP #endif @@ -36,6 +37,18 @@ IMPLEMENT_DYNAMIC_CLASS(wxRichTextHTMLHandler, wxRichTextFileHandler) int wxRichTextHTMLHandler::sm_fileCounter = 1; +wxRichTextHTMLHandler::wxRichTextHTMLHandler(const wxString& name, const wxString& ext, int type) + : wxRichTextFileHandler(name, ext, type), m_buffer(NULL), m_font(false), m_inTable(false) +{ + m_fontSizeMapping.Add(8); + m_fontSizeMapping.Add(10); + m_fontSizeMapping.Add(13); + m_fontSizeMapping.Add(17); + m_fontSizeMapping.Add(22); + m_fontSizeMapping.Add(30); + m_fontSizeMapping.Add(100); +} + /// Can we handle this filename (if using files)? By default, checks the extension. bool wxRichTextHTMLHandler::CanHandle(const wxString& filename) const { @@ -58,10 +71,12 @@ bool wxRichTextHTMLHandler::DoLoadFile(wxRichTextBuffer *WXUNUSED(buffer), wxInp bool wxRichTextHTMLHandler::DoSaveFile(wxRichTextBuffer *buffer, wxOutputStream& stream) { + m_buffer = buffer; + ClearTemporaryImageLocations(); buffer->Defragment(); - + wxTextOutputStream str(stream); wxTextAttrEx currentParaStyle = buffer->GetAttributes(); @@ -70,15 +85,15 @@ bool wxRichTextHTMLHandler::DoSaveFile(wxRichTextBuffer *buffer, wxOutputStream& if ((GetFlags() & wxRICHTEXT_HANDLER_NO_HEADER_FOOTER) == 0) str << wxT("
\n"); - str << wxT("\n");
-
- str << wxString::Format(wxT(""),
- currentParaStyle.GetFont().GetFaceName().c_str(), PtToSize(currentParaStyle.GetFont().GetPointSize()),
- currentParaStyle.GetTextColour().GetAsString(wxC2S_HTML_SYNTAX).c_str());
+ str << wxT("
|
"); if ((GetFlags() & wxRICHTEXT_HANDLER_NO_HEADER_FOOTER) == 0) str << wxT(""); str << wxT("\n"); + m_buffer = NULL; + return true; } -void wxRichTextHTMLHandler::BeginCharacterFormatting(const wxTextAttrEx& currentStyle, const wxTextAttrEx& thisStyle, const wxTextAttrEx& paraStyle, wxOutputStream& stream) +void wxRichTextHTMLHandler::BeginCharacterFormatting(const wxTextAttrEx& currentStyle, const wxTextAttrEx& thisStyle, const wxTextAttrEx& WXUNUSED(paraStyle), wxTextOutputStream& str) { - wxTextOutputStream str(stream); - - // Is the item a bulleted one? - if ( paraStyle.GetBulletStyle() != wxTEXT_ATTR_BULLET_STYLE_NONE ) - { - // Is there any opened list? - if (m_list) - { - // Yes there is - - // Is the item among the previous ones? - // Is the item one of the previous list tag's child items? - if ((paraStyle.GetLeftIndent() == (m_indent + 100)) || (paraStyle.GetLeftIndent() < 100)) - str << wxT("
after every paragraph - if (!m_list) - { - wxTextOutputStream str(stream); - wxString align = GetAlignment(thisStyle); - str << wxString::Format(wxT("
"), align.c_str()); - } - if (thisStyle.HasPageBreak()) { - wxTextOutputStream str(stream); + str << wxT(""); str << wxT("
\n"); + str << wxT("");
}
-}
-void wxRichTextHTMLHandler::NavigateToListPosition(const wxTextAttrEx& thisStyle, wxTextOutputStream& str)
-{
- // indenting an item using an ul/ol tag is equal to inserting 5 x on its left side.
- // so we should start from 100 point left
-
- // Is the second td's left wall of the current indentaion table at the 100+ point-left-side
- // of the item, horizontally?
- if (m_indent + 100 < thisStyle.GetLeftIndent())
+ if (thisStyle.HasLeftIndent())
{
- // yes it is
- LIndent(thisStyle, str);
- m_indent = thisStyle.GetLeftIndent() - 100;
- m_indents.Add( m_indent );
- return;
- }
- // No it isn't
-
- int i = m_indents.size() - 1;
- for (; i > -1; i--)
- {
- //Is the second td's left wall of the current indentaion table at the 100+ point-left-side
- //of the item ?
- if (m_indent + 100 < thisStyle.GetLeftIndent())
+ if (thisStyle.HasBulletStyle())
{
- // Yes it is
- LIndent(thisStyle, str);
- m_indent = thisStyle.GetLeftIndent() - 100;
- m_indents.Add( m_indent );
- break;
+ int indent = thisStyle.GetLeftIndent();
+
+ // Close levels high than this
+ CloseLists(indent, str);
+
+ if (m_indents.GetCount() > 0 && indent == m_indents.Last())
+ {
+ // Same level, no need to start a new list
+ }
+ else if (m_indents.GetCount() == 0 || indent > m_indents.Last())
+ {
+ m_indents.Add(indent);
+
+ wxString tag;
+ int listType = TypeOfList(thisStyle, tag);
+ m_listTypes.Add(listType);
+
+ wxString align = GetAlignment(thisStyle);
+ str << wxString::Format(wxT(" "), align.c_str()); + + str << tag; + } + + str << wxT(" |
"), align.c_str()); - m_indents.RemoveAt(i); + // Use a table + int indentTenthsMM = thisStyle.GetLeftIndent() + thisStyle.GetLeftSubIndent(); + // TODO: convert to pixels + int indentPixels = indentTenthsMM/4; + str << wxString::Format(wxT("
"), indentPixels);
- if (i < 1)
+ OutputFont(thisStyle, str);
+
+ if (thisStyle.GetLeftSubIndent() < 0)
{
- m_indent=0; break;
+ str << SymbolicIndent( - thisStyle.GetLeftSubIndent());
}
- m_indent = m_indents[i-1];
+
+ m_inTable = true;
}
}
-}
-void wxRichTextHTMLHandler::Indent( const wxTextAttrEx& thisStyle, wxTextOutputStream& str )
-{
- //There is no way to indent an item in HTML, but we can use tables.
-
- // Item -> "Hello world"
- // Its Left Indentation -> 100
- // Its Left Sub-Indentation ->40
- // A typical indentation-table for the item will be construct as the following
-
- // 3 x nbsp = 60
- // 2 x nbsp = 40
- // LSI = Left Sub Indent
- // LI = Left Indent - LSI
- //
- // -------------------------------------------
- // | nbsp;|nbsp;nbsp;Hello World |
- // | | | | |
- // | V | V |
- // | --LI-- | --LSI-- |
- // -------------------------------------------
-
- str << wxT("
|