-/** A small helper function
- \author Jean-Marc Lasgouttes (2003)
+/**
+ * \file context.C
+ * This file is part of LyX, the document processor.
+ * Licence details can be found in the file COPYING.
+ *
+ * \author Jean-Marc Lasgouttes
+ *
+ * Full author contact details are available in file CREDITS.
*/
+#include <config.h>
+
#include <iostream>
+#include "support/lstrings.h"
#include "context.h"
+
+namespace lyx {
+
using std::ostream;
using std::endl;
+using std::string;
+
+
+namespace {
+
+void begin_layout(ostream & os, LyXLayout_ptr layout, Font const & font,
+ Font const & normalfont)
+{
+ os << "\n\\begin_layout " << layout->name() << "\n";
+ // FIXME: This is not enough for things like
+ // \\Huge par1 \\par par2
+ output_font_change(os, normalfont, font);
+}
+
+
+void end_layout(ostream & os)
+{
+ os << "\n\\end_layout\n";
+}
+
+
+void begin_deeper(ostream & os)
+{
+ os << "\n\\begin_deeper";
+}
+
+
+void end_deeper(ostream & os)
+{
+ os << "\n\\end_deeper";
+}
+
+}
+
+
+bool operator==(Font const & f1, Font const & f2)
+{
+ return
+ f1.size == f2.size &&
+ f1.family == f2.family &&
+ f1.series == f2.series &&
+ f1.shape == f2.shape;
+}
+
+
+void output_font_change(ostream & os, Font const & oldfont,
+ Font const & newfont)
+{
+ if (oldfont.family != newfont.family)
+ os << "\n\\family " << newfont.family << '\n';
+ if (oldfont.series != newfont.series)
+ os << "\n\\series " << newfont.series << '\n';
+ if (oldfont.shape != newfont.shape)
+ os << "\n\\shape " << newfont.shape << '\n';
+ if (oldfont.size != newfont.size)
+ os << "\n\\size " << newfont.size << '\n';
+}
+
+
+Font Context::normalfont;
+bool Context::empty = true;
+
Context::Context(bool need_layout_,
LyXTextClass const & textclass_,
- LyXLayout_ptr layout_, LyXLayout_ptr parent_layout_)
+ LyXLayout_ptr layout_, LyXLayout_ptr parent_layout_,
+ Font font_)
: need_layout(need_layout_),
need_end_layout(false), need_end_deeper(false),
- textclass(textclass_),
- layout(layout_), parent_layout(parent_layout_)
+ has_item(false), deeper_paragraph(false),
+ new_layout_allowed(true), textclass(textclass_),
+ layout(layout_), parent_layout(parent_layout_),
+ font(font_)
{
if (!layout.get())
layout = textclass.defaultLayout();
}
-void Context::check_end_layout(ostream & os)
+Context::~Context()
{
- if (need_end_layout) {
- os << "\n\\end_layout\n";
- need_end_layout = false;
- }
- if (need_end_deeper) {
- os << "\n\\end_deeper\n";
- need_end_deeper = false;
- }
+ if (!extra_stuff.empty())
+ std::cerr << "Bug: Ignoring extra stuff '" << extra_stuff
+ << '\'' << std::endl;
}
void Context::check_layout(ostream & os)
{
if (need_layout) {
- if (parent_layout->isEnvironment()) {
- if (need_end_deeper) {
- // no need to have \end_deeper \begin_deeper
-// FIXME: This does not work because \par already calls check_end_layout
- need_end_deeper = false;
- check_end_layout(os);
+ check_end_layout(os);
+
+ // are we in a list-like environment?
+ if (layout->isEnvironment()
+ && layout->latextype != LATEX_ENVIRONMENT) {
+ // A list-like environment
+ if (has_item) {
+ // a new item. If we had a standard
+ // paragraph before, we have to end it.
+ if (deeper_paragraph) {
+ end_deeper(os);
+ deeper_paragraph = false;
+ }
+ begin_layout(os, layout, font, normalfont);
+ has_item = false;
} else {
- check_end_layout(os);
- os << "\n\\begin_deeper\n";
- need_end_deeper = true;
+ // a standard paragraph in an
+ // enumeration. We have to recognize
+ // that this may require a begin_deeper.
+ if (!deeper_paragraph)
+ begin_deeper(os);
+ begin_layout(os, textclass.defaultLayout(),
+ font, normalfont);
+ deeper_paragraph = true;
}
- } else
- check_end_layout(os);
-
- os << "\n\\begin_layout " << layout->name() << "\n\n";
+ } else {
+ // No list-like environment
+ begin_layout(os, layout, font, normalfont);
+ }
+ need_layout = false;
need_end_layout = true;
- need_layout=false;
if (!extra_stuff.empty()) {
os << extra_stuff;
extra_stuff.erase();
}
+ os << "\n";
+ empty = false;
+ }
+}
+
+
+void Context::check_end_layout(ostream & os)
+{
+ if (need_end_layout) {
+ end_layout(os);
+ need_end_layout = false;
+ }
+}
+
+
+void Context::check_deeper(ostream & os)
+{
+ if (parent_layout->isEnvironment()) {
+ // We start a nested environment.
+ // We need to increase the depth.
+ if (need_end_deeper) {
+ // no need to have \end_deeper \begin_deeper
+ need_end_deeper = false;
+ } else {
+ begin_deeper(os);
+ need_end_deeper = true;
+ }
+ } else
+ check_end_deeper(os);
+}
+
+
+void Context::check_end_deeper(ostream & os)
+{
+ if (need_end_deeper) {
+ end_deeper(os);
+ need_end_deeper = false;
+ }
+ if (deeper_paragraph) {
+ end_deeper(os);
+ deeper_paragraph = false;
}
}
+void Context::set_item()
+{
+ need_layout = true;
+ has_item = true;
+}
+
+
+void Context::new_paragraph(ostream & os)
+{
+ check_end_layout(os);
+ need_layout = true;
+}
+
+
+void Context::add_extra_stuff(std::string const & stuff)
+{
+ if (!lyx::support::contains(extra_stuff, stuff))
+ extra_stuff += stuff;
+}
+
+
void Context::dump(ostream & os, string const & desc) const
{
- os << desc <<" [";
+ os << "\n" << desc <<" [";
if (need_layout)
os << "need_layout ";
if (need_end_layout)
os << "need_end_layout ";
+ if (need_end_deeper)
+ os << "need_end_deeper ";
+ if (has_item)
+ os << "has_item ";
+ if (deeper_paragraph)
+ os << "deeper_paragraph ";
+ if (new_layout_allowed)
+ os << "new_layout_allowed ";
if (!extra_stuff.empty())
os << "extrastuff=[" << extra_stuff << "] ";
- os << "layout=" << layout->name();
- os << " parent_layout=" << parent_layout->name() << "]" << endl;
+ os << "textclass=" << textclass.name()
+ << " layout=" << layout->name()
+ << " parent_layout=" << parent_layout->name() << "] font=["
+ << font.size << ' ' << font.family << ' ' << font.series << ' '
+ << font.shape << ']' << endl;
}
+
+
+} // namespace lyx