lyx_mirror/src/tex2lyx/Context.cpp
Juergen Spitzmueller e51265b5f3 tex2lyx: More work on beamer support
* Implement list item overlay support (\item<arg>)
* Implement itemcommand support (e.g., \overprint<arg>)
* Fix general list argument placement

Part of: #11068
2018-03-14 08:40:47 +01:00

261 lines
5.5 KiB
C++

/**
* \file Context.cpp
* 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 "Context.h"
#include "Layout.h"
#include "support/lstrings.h"
#include <iostream>
using namespace std;
using namespace lyx::support;
namespace lyx {
namespace {
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";
}
} // namespace
bool operator==(TeXFont const & f1, TeXFont const & f2)
{
return
f1.size == f2.size &&
f1.family == f2.family &&
f1.series == f2.series &&
f1.shape == f2.shape &&
f1.language == f2.language;
}
void output_font_change(ostream & os, TeXFont const & oldfont,
TeXFont 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';
if (oldfont.language != newfont.language)
os << "\n\\lang " << newfont.language << '\n';
}
TeXFont Context::normalfont;
bool Context::empty = true;
Context::Context(bool need_layout_,
TeX2LyXDocClass const & textclass_,
Layout const * layout_, Layout const * parent_layout_,
TeXFont const & font_)
: need_layout(need_layout_),
need_end_layout(false), need_end_deeper(false),
has_item(false), deeper_paragraph(false),
new_layout_allowed(true), merging_hyphens_allowed(true),
textclass(textclass_),
layout(layout_), parent_layout(parent_layout_),
font(font_)
{
if (!layout)
layout = &textclass.defaultLayout();
if (!parent_layout)
parent_layout = &textclass.defaultLayout();
}
Context::~Context()
{
if (!par_extra_stuff.empty())
cerr << "Bug: Ignoring par-level extra stuff '"
<< par_extra_stuff << '\'' << endl;
}
void Context::begin_layout(ostream & os, Layout const * const & l)
{
os << "\n\\begin_layout " << to_utf8(l->name()) << "\n";
if (!extra_stuff.empty()) {
os << extra_stuff;
}
if (!par_extra_stuff.empty()) {
os << par_extra_stuff;
par_extra_stuff.erase();
}
// FIXME: This is not enough for things like
// \\Huge par1 \\par par2
// FIXME: If the document language is not english this outputs a
// superflous language change. Fortunately this is only file format
// bloat and does not change the TeX export of LyX.
output_font_change(os, normalfont, font);
}
void Context::check_layout(ostream & os)
{
if (need_layout) {
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);
has_item = false;
} else {
// 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());
deeper_paragraph = true;
}
} else {
// No list-like environment
begin_layout(os, layout);
}
need_layout = false;
need_end_layout = true;
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(string const & stuff)
{
if (!contains(extra_stuff, stuff))
extra_stuff += stuff;
}
void Context::add_par_extra_stuff(string const & stuff)
{
if (!contains(par_extra_stuff, stuff))
par_extra_stuff += stuff;
}
void Context::dump(ostream & os, string const & desc) const
{
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 (merging_hyphens_allowed)
os << "merging_hyphens_allowed ";
if (!extra_stuff.empty())
os << "extrastuff=[" << extra_stuff << "] ";
if (!par_extra_stuff.empty())
os << "parextrastuff=[" << par_extra_stuff << "] ";
if (!list_extra_stuff.empty())
os << "listextrastuff=[" << list_extra_stuff << "] ";
os << "textclass=" << textclass.name()
<< " layout=" << to_utf8(layout->name())
<< " parent_layout=" << to_utf8(parent_layout->name()) << "] font=["
<< font.size << ' ' << font.family << ' ' << font.series << ' '
<< font.shape << ']' << endl;
}
} // namespace lyx