lyx_mirror/src/insets/InsetRef.cpp
Richard Heck 996505c2ae This patch introduces an optional argument to Buffer::updateLabels(), so
as to allow us to call the routine when we are preparing for output and
so to do certain things we might not want to do every time. 

This is an abuse of updateLabels(), in a way, but updateLabels() long
ago became the general recurse-through-the-Buffer routine, and to
implement the sort of thing I want to do here in validate(), say, much
of the code in updateLabels()---in particular, the counter-update
code---would have to be duplicated. So I believe this is the best, and
easiest, way to go.

Actual use of the new argument will follow.


git-svn-id: svn://svn.lyx.org/lyx/lyx-devel/trunk@32318 a592a061-630c-0410-9148-cb99ea01b6c8
2009-12-04 03:03:38 +00:00

211 lines
5.1 KiB
C++

/**
* \file InsetRef.cpp
* This file is part of LyX, the document processor.
* Licence details can be found in the file COPYING.
*
* \author José Matos
*
* Full author contact details are available in file CREDITS.
*/
#include <config.h>
#include "InsetRef.h"
#include "Buffer.h"
#include "Cursor.h"
#include "DispatchResult.h"
#include "FuncRequest.h"
#include "LaTeXFeatures.h"
#include "LyXFunc.h"
#include "OutputParams.h"
#include "output_xhtml.h"
#include "ParIterator.h"
#include "sgml.h"
#include "TocBackend.h"
#include "support/docstream.h"
#include "support/gettext.h"
#include "support/lstrings.h"
using namespace lyx::support;
using namespace std;
namespace lyx {
InsetRef::InsetRef(Buffer * buf, InsetCommandParams const & p)
: InsetCommand(buf, p, "ref"), isLatex(buf->isLatex())
{}
InsetRef::InsetRef(InsetRef const & ir)
: InsetCommand(ir), isLatex(ir.isLatex)
{}
bool InsetRef::isCompatibleCommand(string const & s) {
//FIXME This is likely not the best way to handle this.
//But this stuff is hardcoded elsewhere already.
return s == "ref"
|| s == "pageref"
|| s == "vref"
|| s == "vpageref"
|| s == "prettyref"
|| s == "eqref";
}
ParamInfo const & InsetRef::findInfo(string const & /* cmdName */)
{
static ParamInfo param_info_;
if (param_info_.empty()) {
param_info_.add("name", ParamInfo::LATEX_OPTIONAL);
param_info_.add("reference", ParamInfo::LATEX_REQUIRED);
}
return param_info_;
}
docstring InsetRef::screenLabel() const
{
return screen_label_;
}
int InsetRef::latex(odocstream & os, OutputParams const &) const
{
// We don't want to output p_["name"], since that is only used
// in docbook. So we construct new params, without it, and use that.
InsetCommandParams p(REF_CODE, getCmdName());
p["reference"] = getParam("reference");
os << escape(p.getCommand());
return 0;
}
int InsetRef::plaintext(odocstream & os, OutputParams const &) const
{
docstring const str = getParam("reference");
os << '[' << str << ']';
return 2 + str.size();
}
int InsetRef::docbook(odocstream & os, OutputParams const & runparams) const
{
docstring const & name = getParam("name");
if (name.empty()) {
if (runparams.flavor == OutputParams::XML) {
os << "<xref linkend=\""
<< sgml::cleanID(buffer(), runparams, getParam("reference"))
<< "\" />";
} else {
os << "<xref linkend=\""
<< sgml::cleanID(buffer(), runparams, getParam("reference"))
<< "\">";
}
} else {
os << "<link linkend=\""
<< sgml::cleanID(buffer(), runparams, getParam("reference"))
<< "\">"
<< getParam("name")
<< "</link>";
}
return 0;
}
docstring InsetRef::xhtml(XHTMLStream & xs, OutputParams const &) const
{
// FIXME What we'd really like to do is to be able to output some
// appropriate sort of text here. But to do that, we need to associate
// some sort of counter with the label, and we don't have that yet.
docstring const ref = html::cleanAttr(getParam("reference"));
string const attr = "href=\"#" + to_utf8(ref) + "\"";
xs << StartTag("a", attr);
xs << ref;
xs << EndTag("a");
return docstring();
}
void InsetRef::tocString(odocstream & os) const
{
plaintext(os, OutputParams(0));
}
void InsetRef::updateLabels(ParIterator const & it, bool)
{
docstring const & label = getParam("reference");
// register this inset into the buffer reference cache.
buffer().references(label).push_back(make_pair(this, it));
for (int i = 0; !types[i].latex_name.empty(); ++i) {
if (getCmdName() == types[i].latex_name) {
screen_label_ = _(types[i].short_gui_name);
break;
}
}
screen_label_ += label;
if (!isLatex && !getParam("name").empty()) {
screen_label_ += "||";
screen_label_ += getParam("name");
}
}
void InsetRef::addToToc(DocIterator const & cpit)
{
docstring const & label = getParam("reference");
if (buffer().insetLabel(label))
// This InsetRef has already been taken care of in InsetLabel::addToToc().
return;
// It seems that this reference does not point to any valid label.
screen_label_ = _("BROKEN: ") + screen_label_;
Toc & toc = buffer().tocBackend().toc("label");
toc.push_back(TocItem(cpit, 0, screen_label_));
}
void InsetRef::validate(LaTeXFeatures & features) const
{
if (getCmdName() == "vref" || getCmdName() == "vpageref")
features.require("varioref");
else if (getCmdName() == "prettyref")
features.require("prettyref");
else if (getCmdName() == "eqref")
features.require("amsmath");
}
InsetRef::type_info InsetRef::types[] = {
{ "ref", N_("Standard"), N_("Ref: ")},
{ "eqref", N_("Equation"), N_("EqRef: ")},
{ "pageref", N_("Page Number"), N_("Page: ")},
{ "vpageref", N_("Textual Page Number"), N_("TextPage: ")},
{ "vref", N_("Standard+Textual Page"), N_("Ref+Text: ")},
{ "prettyref", N_("PrettyRef"), N_("FormatRef: ")},
{ "", "", "" }
};
int InsetRef::getType(string const & name)
{
for (int i = 0; !types[i].latex_name.empty(); ++i)
if (name == types[i].latex_name)
return i;
return 0;
}
string const & InsetRef::getName(int type)
{
return types[type].latex_name;
}
} // namespace lyx