]> matita.cs.unibo.it Git - helm.git/blob - helm/DEVEL/mathml_editor/src/CMathMLFactoryXSLT.cc
When the stylesheet from TML to MathML generated a document without a root
[helm.git] / helm / DEVEL / mathml_editor / src / CMathMLFactoryXSLT.cc
1 /* This file is part of EdiTeX, an editor of mathematical
2  * expressions based on TeX syntax.
3  * 
4  * Copyright (C) 2002-2003 Luca Padovani <lpadovan@cs.unibo.it>,
5  *                    2003 Paolo Marinelli <pmarinel@cs.unibo.it>.
6  *
7  * This library is free software; you can redistribute it and/or
8  * modify it under the terms of the GNU Lesser General Public
9  * License as published by the Free Software Foundation; either
10  * version 2.1 of the License, or (at your option) any later version.
11  *
12  * This library is distributed in the hope that it will be useful,
13  * but WITHOUT ANY WARRANTY; without even the implied warranty of
14  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
15  * Lesser General Public License for more details.
16  *
17  * You should have received a copy of the GNU Lesser General Public
18  * License along with this library; if not, write to the Free Software
19  * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA
20  *
21  * For more information, please visit the project's home page
22  * http://helm.cs.unibo.it/editex/
23  * or send an email to <lpadovan@cs.unibo.it>
24  */
25
26 #include "dom.hh"
27 #include "TNode.hh"
28 #include "ALogger.hh"
29 #include "TDocument.hh"
30 #include "CMathMLFactoryXSLT.hh"
31 #include "AMathMLConsumer.hh"
32
33 CMathMLFactoryXSLT::CMathMLFactoryXSLT(ALogger& l, const DOMX::XSLTStylesheet& s)
34   : AMathMLFactory(l), style(s)
35 {
36   DOM::DOMImplementation di;
37   DOM::DocumentType dt;
38   result = di.createDocument(MATHML_NS_URI, "m:math", dt);
39 }
40
41 void
42 CMathMLFactoryXSLT::documentModified(TDocument& doc)
43 {
44   std::vector< std::pair<DOM::GdomeString, DOM::GdomeString> > dirtyId;
45   if (TNode dirty = doc.dirtyNode())
46     if (result.get_documentElement().hasAttribute("xref"))
47       dirtyId.push_back(std::make_pair(DOM::GdomeString("id"),
48                                        DOM::GdomeString("'" + std::string(dirty["id"]) + "'")));
49   DOM::Document res = style.apply(doc.document(), dirtyId);
50   assert(res);
51   style.save(doc.document(), stdout);
52
53   if (DOM::Element root = res.get_documentElement())
54           {
55       if (result.get_documentElement().hasAttribute("xref"))
56         {
57           bool ok = subst(result.get_documentElement(), root.getAttribute("xref"), result.importNode(root, true));
58           assert(ok);
59           doc.clearDirty();
60         }
61       else
62         result.replaceChild(result.importNode(root, true), result.get_documentElement());
63           }
64         else
65                 {
66                         // Something wrong happened while applying the stylesheet.
67                         DOM::Element root = result.get_documentElement();
68                         DOM::Node p = root.get_firstChild();
69                         while (p) {
70                                 DOM::Node next = p.get_nextSibling();
71                                 root.removeChild(p);
72                                 p = next;
73                         }
74       logger.error("The stylesheet produced an empty document");
75     }
76
77   style.save(result, stdout);
78 }
79
80 bool
81 CMathMLFactoryXSLT::subst(const DOM::Element& e1, const DOM::GdomeString& id, const DOM::Element& e2)
82 {
83   assert(e1);
84   assert(e2);
85   if (e1.getAttribute("xref") == id)
86     {
87       DOM::Node parent = e1.get_parentNode();
88       assert(parent);
89       parent.replaceChild(e2, e1);
90       return true;
91     }
92   else
93     {
94       DOM::Node p = e1.get_firstChild();
95       while (p)
96         {
97           while (p && p.get_nodeType() != DOM::Node::ELEMENT_NODE) p = p.get_nextSibling();
98           if (p)
99             if (subst(p, id, e2)) return true;
100             else p = p.get_nextSibling();
101         }
102       return false;
103     }
104 }
105