Also track local language switches

Until now this was not done for essentially two reasons. The first
one is that local switches are used for short text insertions, so that
they are unlikely crossing environment boundaries. The second one
is that if we have to close a language at the end of an environment
we would be missing the right termination command. As this last
issue can be overcome by simply storing in the stack the current
nest level with a sign denoting the kind of switch, there is no
reason anymore not to track also local languages switches.
Also factor out some commonly used constructs in order to improve
readability.
This commit is contained in:
Enrico Forestieri 2016-09-27 23:00:29 +02:00
parent e40fa86a97
commit 7e3494c293
3 changed files with 54 additions and 53 deletions

View File

@ -287,7 +287,7 @@ int Font::latexWriteStartChanges(odocstream & os, BufferParams const & bparams,
tmp += "{"; tmp += "{";
os << from_ascii(tmp); os << from_ascii(tmp);
count += tmp.length(); count += tmp.length();
pushPolyglossiaLang(language()->polyglossia()); pushPolyglossiaLang(language()->polyglossia(), true);
} else if (language()->encoding()->package() != Encoding::CJK) { } else if (language()->encoding()->package() != Encoding::CJK) {
os << '{'; os << '{';
count += 1; count += 1;

View File

@ -107,7 +107,25 @@ bool atSameLastLangSwitchDepth(OutputState const * state)
return state->lang_switch_depth_.size() == 0 return state->lang_switch_depth_.size() == 0
? true ? true
: state->lang_switch_depth_.top() == state->nest_level_; : abs(state->lang_switch_depth_.top()) == state->nest_level_;
}
bool isLocalSwitch(OutputState const * state)
{
// Return true if the language was opened by the \text<lang> command.
return state->lang_switch_depth_.size()
&& state->lang_switch_depth_.top() < 0;
}
bool langOpenedAtThisLevel(OutputState const * state)
{
// Return true if the language was opened at the current nesting level.
return state->lang_switch_depth_.size()
&& abs(state->lang_switch_depth_.top()) == state->nest_level_;
} }
@ -204,10 +222,8 @@ static TeXEnvironmentData prepareEnvironment(Buffer const & buf,
prev_par_lang)) prev_par_lang))
// the '%' is necessary to prevent unwanted whitespace // the '%' is necessary to prevent unwanted whitespace
<< "%\n"; << "%\n";
if (use_polyglossia) { if (use_polyglossia)
state->lang_switch_depth_.pop(); popPolyglossiaLang();
state->open_polyglossia_lang_.pop();
}
} }
// If no language was explicitly opened and we are using // If no language was explicitly opened and we are using
@ -229,10 +245,8 @@ static TeXEnvironmentData prepareEnvironment(Buffer const & buf,
os << bc; os << bc;
// the '%' is necessary to prevent unwanted whitespace // the '%' is necessary to prevent unwanted whitespace
os << "%\n"; os << "%\n";
if (use_polyglossia) { if (use_polyglossia)
state->lang_switch_depth_.push(state->nest_level_); pushPolyglossiaLang(par_lang);
state->open_polyglossia_lang_.push(par_lang);
}
} }
} }
@ -298,13 +312,15 @@ static void finishEnvironment(otexstream & os, OutputParams const & runparams,
os << breakln; os << breakln;
// Close any polyglossia language opened at this nest level // Close any polyglossia language opened at this nest level
if (runparams.use_polyglossia) { if (runparams.use_polyglossia) {
OutputState * state = getOutputState(); while (langOpenedAtThisLevel(state)) {
while (state->lang_switch_depth_.size() if (isLocalSwitch(state)) {
&& state->lang_switch_depth_.top() == state->nest_level_) { os << "}";
os << "\\end{" << openPolyglossiaLang(state) } else {
<< "}%\n"; os << "\\end{"
state->lang_switch_depth_.pop(); << openPolyglossiaLang(state)
state->open_polyglossia_lang_.pop(); << "}%\n";
}
popPolyglossiaLang();
} }
} }
state->nest_level_ -= 1; state->nest_level_ -= 1;
@ -486,11 +502,12 @@ void getArgInsets(otexstream & os, OutputParams const & runparams, Layout::LaTeX
} // namespace anon } // namespace anon
void pushPolyglossiaLang(string const & lang_name) void pushPolyglossiaLang(string const & lang_name, bool localswitch)
{ {
OutputState * state = getOutputState(); OutputState * state = getOutputState();
state->lang_switch_depth_.push(state->nest_level_); int nest_level = localswitch ? -state->nest_level_ : state->nest_level_;
state->lang_switch_depth_.push(nest_level);
state->open_polyglossia_lang_.push(lang_name); state->open_polyglossia_lang_.push(lang_name);
} }
@ -790,10 +807,8 @@ void TeXOnePar(Buffer const & buf,
"$$lang", "$$lang",
prev_lang)) prev_lang))
<< lang_command_termination; << lang_command_termination;
if (use_polyglossia && !localswitch) { if (use_polyglossia)
state->lang_switch_depth_.pop(); popPolyglossiaLang();
state->open_polyglossia_lang_.pop();
}
} }
// We need to open a new language if we couldn't close the previous // We need to open a new language if we couldn't close the previous
@ -850,10 +865,8 @@ void TeXOnePar(Buffer const & buf,
: subst(lang_begin_command, "$$lang", par_lang); : subst(lang_begin_command, "$$lang", par_lang);
os << bc; os << bc;
os << lang_command_termination; os << lang_command_termination;
if (use_polyglossia && !localswitch) { if (use_polyglossia)
state->lang_switch_depth_.push(state->nest_level_); pushPolyglossiaLang(par_lang, localswitch);
state->open_polyglossia_lang_.push(par_lang);
}
} }
} }
} }
@ -912,10 +925,8 @@ void TeXOnePar(Buffer const & buf,
"$$lang", "$$lang",
par_lang)) par_lang))
<< lang_command_termination; << lang_command_termination;
if (use_polyglossia && !localswitch) { if (use_polyglossia)
state->lang_switch_depth_.push(state->nest_level_); pushPolyglossiaLang(par_lang, localswitch);
state->open_polyglossia_lang_.push(par_lang);
}
} }
runparams.encoding = encoding; runparams.encoding = encoding;
} }
@ -1081,11 +1092,8 @@ void TeXOnePar(Buffer const & buf,
os << bc; os << bc;
pending_newline = !localswitch; pending_newline = !localswitch;
unskip_newline = !localswitch; unskip_newline = !localswitch;
if (use_polyglossia && !localswitch) { if (use_polyglossia)
state->lang_switch_depth_.push( pushPolyglossiaLang(current_lang, localswitch);
state->nest_level_);
state->open_polyglossia_lang_.push(current_lang);
}
} }
} else if (!par_lang.empty()) { } else if (!par_lang.empty()) {
// If we are in an environment, we have to close the "outer" language afterwards // If we are in an environment, we have to close the "outer" language afterwards
@ -1110,10 +1118,8 @@ void TeXOnePar(Buffer const & buf,
par_lang)); par_lang));
pending_newline = !localswitch; pending_newline = !localswitch;
unskip_newline = !localswitch; unskip_newline = !localswitch;
if (use_polyglossia && !localswitch) { if (use_polyglossia)
state->lang_switch_depth_.pop(); popPolyglossiaLang();
state->open_polyglossia_lang_.pop();
}
} }
} }
} }
@ -1291,10 +1297,8 @@ void latexParagraphs(Buffer const & buf,
: subst(lang_begin_command, "$$lang", mainlang); : subst(lang_begin_command, "$$lang", mainlang);
os << bc; os << bc;
os << '\n'; os << '\n';
if (runparams.use_polyglossia) { if (runparams.use_polyglossia)
state->lang_switch_depth_.push(state->nest_level_); pushPolyglossiaLang(mainlang);
state->open_polyglossia_lang_.push(mainlang);
}
} }
ParagraphList const & paragraphs = text.paragraphs(); ParagraphList const & paragraphs = text.paragraphs();
@ -1399,10 +1403,8 @@ void latexParagraphs(Buffer const & buf,
"$$lang", "$$lang",
mainlang)) mainlang))
<< '\n'; << '\n';
if (runparams.use_polyglossia) { if (runparams.use_polyglossia)
state->lang_switch_depth_.pop(); popPolyglossiaLang();
state->open_polyglossia_lang_.pop();
}
} }
// If the last paragraph is an environment, we'll have to close // If the last paragraph is an environment, we'll have to close
@ -1418,10 +1420,8 @@ void latexParagraphs(Buffer const & buf,
"$$lang", "$$lang",
pol_lang)) pol_lang))
<< '\n'; << '\n';
if (runparams.use_polyglossia) { if (runparams.use_polyglossia)
state->lang_switch_depth_.pop(); popPolyglossiaLang();
state->open_polyglossia_lang_.pop();
}
} }
// reset inherited encoding // reset inherited encoding

View File

@ -30,8 +30,9 @@ class OutputParams;
class Text; class Text;
/** Register a language switch when using polyglossia. /** Register a language switch when using polyglossia.
Set \p localswitch to true if the \text<lang> command is used.
*/ */
void pushPolyglossiaLang(std::string const & lang); void pushPolyglossiaLang(std::string const & lang, bool localswitch = false);
/** Unregister the last language switch when using polyglossia. /** Unregister the last language switch when using polyglossia.
*/ */