]> git.lyx.org Git - features.git/blobdiff - src/output_xhtml.cpp
Add button to stop processing
[features.git] / src / output_xhtml.cpp
index dcc3f69c333c438a451452d17c60b38a1110564a..f1bcfb879b3cfba5b579fdde04d05b65bc9c41a8 100644 (file)
@@ -3,7 +3,7 @@
  * This file is part of LyX, the document processor.
  * Licence details can be found in the file COPYING.
  *
- * \author Richard Heck
+ * \author Richard Kimberly Heck
  *
  * This code is based upon output_docbook.cpp
  *
 #include "output_xhtml.h"
 
 #include "Buffer.h"
-#include "buffer_funcs.h"
 #include "BufferParams.h"
 #include "Counters.h"
 #include "Font.h"
 #include "Layout.h"
-#include "OutputParams.h"
 #include "Paragraph.h"
 #include "ParagraphList.h"
 #include "ParagraphParameters.h"
 #include "Text.h"
 #include "TextClass.h"
 
-#include "support/convert.h"
-#include "support/debug.h"
 #include "support/lassert.h"
-#include "support/lstrings.h"
-#include "support/textutils.h"
 
-#include <vector>
+#include <stack>
+#include <iostream>
 
 // Uncomment to activate debugging code.
 // #define XHTML_DEBUG
@@ -45,7 +40,7 @@ using namespace lyx::support;
 namespace lyx {
 
 
-docstring const & fontToHtmlTag(xml::FontTypes type)
+docstring fontToHtmlTag(xml::FontTypes type)
 {
     switch(type) {
         case xml::FontTypes::FT_EMPH:
@@ -88,70 +83,66 @@ docstring const & fontToHtmlTag(xml::FontTypes type)
 }
 
 
-namespace {
-
-string fontToAttribute(xml::FontTypes type)
+docstring fontToHtmlAttribute(xml::FontTypes type)
 {
        switch(type) {
        case xml::FontTypes::FT_EMPH:
        case xml::FontTypes::FT_BOLD:
-               return "";
+               return from_ascii("");
        case xml::FontTypes::FT_NOUN:
-               return "class='lyxnoun'";
+               return from_ascii("class='lyxnoun'");
        case xml::FontTypes::FT_UBAR:
-               return "";
+               return from_ascii("");
        case xml::FontTypes::FT_DBAR:
-               return "class='dline'";
+               return from_ascii("class='dline'");
        case xml::FontTypes::FT_XOUT:
        case xml::FontTypes::FT_SOUT:
-               return "class='strikeout'";
+               return from_ascii("class='strikeout'");
        case xml::FontTypes::FT_WAVE:
-               return "class='wline'";
+               return from_ascii("class='wline'");
        case xml::FontTypes::FT_ITALIC:
-               return "";
+               return from_ascii("");
        case xml::FontTypes::FT_UPRIGHT:
-               return "style='font-style:normal;'";
+               return from_ascii("style='font-style:normal;'");
        case xml::FontTypes::FT_SLANTED:
-               return "style='font-style:oblique;'";
+               return from_ascii("style='font-style:oblique;'");
        case xml::FontTypes::FT_SMALLCAPS:
-               return "style='font-variant:small-caps;'";
+               return from_ascii("style='font-variant:small-caps;'");
        case xml::FontTypes::FT_ROMAN:
-               return "style='font-family:serif;'";
+               return from_ascii("style='font-family:serif;'");
        case xml::FontTypes::FT_SANS:
-               return "style='font-family:sans-serif;'";
+               return from_ascii("style='font-family:sans-serif;'");
        case xml::FontTypes::FT_TYPE:
-               return "style='font-family:monospace;'";
+               return from_ascii("style='font-family:monospace;'");
        case xml::FontTypes::FT_SIZE_TINY:
        case xml::FontTypes::FT_SIZE_SCRIPT:
        case xml::FontTypes::FT_SIZE_FOOTNOTE:
-               return "style='font-size:x-small;'";
+               return from_ascii("style='font-size:x-small;'");
        case xml::FontTypes::FT_SIZE_SMALL:
-               return "style='font-size:small;'";
+               return from_ascii("style='font-size:small;'");
        case xml::FontTypes::FT_SIZE_NORMAL:
-               return "style='font-size:normal;'";
+               return from_ascii("style='font-size:normal;'");
        case xml::FontTypes::FT_SIZE_LARGE:
-               return "style='font-size:large;'";
+               return from_ascii("style='font-size:large;'");
        case xml::FontTypes::FT_SIZE_LARGER:
        case xml::FontTypes::FT_SIZE_LARGEST:
-               return "style='font-size:x-large;'";
+               return from_ascii("style='font-size:x-large;'");
        case xml::FontTypes::FT_SIZE_HUGE:
        case xml::FontTypes::FT_SIZE_HUGER:
-               return "style='font-size:xx-large;'";
+               return from_ascii("style='font-size:xx-large;'");
        case xml::FontTypes::FT_SIZE_INCREASE:
-               return "style='font-size:larger;'";
+               return from_ascii("style='font-size:larger;'");
        case xml::FontTypes::FT_SIZE_DECREASE:
-               return "style='font-size:smaller;'";
+               return from_ascii("style='font-size:smaller;'");
        }
        // kill warning
-       return "";
+       return from_ascii("");
 }
 
-} // end anonymous namespace
-
 
 xml::FontTag xhtmlStartFontTag(xml::FontTypes type)
 {
-       return xml::FontTag(fontToHtmlTag(type), from_utf8(fontToAttribute(type)), type);
+       return xml::FontTag(fontToHtmlTag(type), fontToHtmlAttribute(type), type);
 }
 
 
@@ -165,15 +156,18 @@ namespace {
 // convenience functions
 
 inline void openParTag(XMLStream & xs, Layout const & lay,
-                       std::string parlabel)
+                       const std::string & parlabel)
 {
-       xs << xml::ParTag(lay.htmltag(), lay.htmlattr(), parlabel);
+       string attrs = lay.htmlattr();
+       if (!parlabel.empty())
+               attrs += " id='" + parlabel + "'";
+       xs << xml::ParTag(lay.htmltag(), attrs);
 }
 
 
 void openParTag(XMLStream & xs, Layout const & lay,
                 ParagraphParameters const & params,
-                std::string parlabel)
+                const std::string & parlabel)
 {
        // FIXME Are there other things we should handle here?
        string const align = alignmentToCSS(params.align());
@@ -182,7 +176,9 @@ void openParTag(XMLStream & xs, Layout const & lay,
                return;
        }
        string attrs = lay.htmlattr() + " style='text-align: " + align + ";'";
-       xs << xml::ParTag(lay.htmltag(), attrs, parlabel);
+       if (!parlabel.empty())
+               attrs += " id='" + parlabel + "'";
+       xs << xml::ParTag(lay.htmltag(), attrs);
 }
 
 
@@ -589,6 +585,8 @@ void xhtmlParagraphs(Text const & text,
        ParagraphList::const_iterator const pend =
                (epit == (int) paragraphs.size()) ?
                        paragraphs.end() : paragraphs.iterator_at(epit);
+       std::stack<int> headerLevels;
+
        while (bpit < epit) {
                ParagraphList::const_iterator par = paragraphs.iterator_at(bpit);
                if (par->params().startOfAppendix()) {
@@ -606,6 +604,35 @@ void xhtmlParagraphs(Text const & text,
                ParagraphList::const_iterator const lastpar = par;
                ParagraphList::const_iterator send;
 
+               // Think about adding <section> and/or </section>s.
+               // Document title is not in Sectioning, but rather in FrontMatter, so that it does not need to be taken
+               // into account.
+               if (style.category() == from_utf8("Sectioning")) {
+                       int level = style.toclevel;
+
+                       // Need to close a previous section if it has the same level or a higher one (close <section> if opening a
+                       // <h2> after a <h2>, <h3>, <h4>, <h5> or <h6>). More examples:
+                       //   - current: h2; back: h1; do not close any <section>
+                       //   - current: h1; back: h2; close two <section> (first the <h2>, then the <h1>, so a new <h1> can come)
+                       while (!headerLevels.empty() && level <= headerLevels.top()) {
+                               // Output the tag only if it corresponds to a legit section.
+                               int stackLevel = headerLevels.top();
+                               if (stackLevel != Layout::NOT_IN_TOC) {
+                                       xs << xml::EndTag("section");
+                                       xs << xml::CR();
+                               }
+                               headerLevels.pop();
+                       }
+
+                       // Open the new section: first push it onto the stack, then output it in XHTML.
+                       headerLevels.push(level);
+                       // Some sectioning-like elements should not be output (such as FrontMatter).
+                       if (level != Layout::NOT_IN_TOC ) {
+                               xs << xml::StartTag("section");
+                               xs << xml::CR();
+                       }
+               }
+
                switch (style.latextype) {
                case LATEX_COMMAND: {
                        // The files with which we are working never have more than
@@ -642,6 +669,14 @@ void xhtmlParagraphs(Text const & text,
                }
                bpit += distance(lastpar, par);
        }
+
+       // If need be, close <section>s, but only at the end of the document (otherwise, dealt with at the beginning
+       // of the loop).
+       while (!headerLevels.empty() && headerLevels.top() != Layout::NOT_IN_TOC) {
+               headerLevels.pop();
+               xs << xml::EndTag("section");
+               xs << xml::CR();
+       }
 }