2015-09-02 19:53:13 +01:00
|
|
|
/**
|
|
|
|
* \file InsetCaptionable.cpp
|
|
|
|
* This file is part of LyX, the document processor.
|
|
|
|
* Licence details can be found in the file COPYING.
|
|
|
|
*
|
|
|
|
* \author Alejandro Aguilar Sierra
|
|
|
|
* \author Jürgen Vigna
|
|
|
|
* \author Lars Gullik Bjønnes
|
|
|
|
* \author Guillaume Munch
|
|
|
|
*
|
|
|
|
* Full author contact details are available in file CREDITS.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <config.h>
|
|
|
|
|
2020-11-15 03:03:16 +01:00
|
|
|
#include "InsetBox.h"
|
2015-09-02 19:53:13 +01:00
|
|
|
#include "InsetCaptionable.h"
|
2016-09-24 00:49:00 +02:00
|
|
|
#include "InsetCaption.h"
|
2020-11-15 03:03:16 +01:00
|
|
|
#include "InsetLabel.h"
|
2016-09-24 00:49:00 +02:00
|
|
|
|
2015-09-02 19:53:13 +01:00
|
|
|
#include "Buffer.h"
|
|
|
|
#include "BufferParams.h"
|
|
|
|
#include "BufferView.h"
|
|
|
|
#include "FloatList.h"
|
2016-09-24 00:49:00 +02:00
|
|
|
#include "InsetList.h"
|
|
|
|
#include "output_xhtml.h"
|
2015-09-02 19:53:13 +01:00
|
|
|
#include "TextClass.h"
|
|
|
|
#include "TocBackend.h"
|
|
|
|
|
2016-09-24 00:49:00 +02:00
|
|
|
#include "support/docstream.h"
|
|
|
|
|
2015-09-02 19:53:13 +01:00
|
|
|
using namespace std;
|
|
|
|
|
|
|
|
|
|
|
|
namespace lyx {
|
|
|
|
|
|
|
|
|
|
|
|
void InsetCaptionable::setCaptionType(std::string const & type)
|
|
|
|
{
|
|
|
|
caption_type_ = type.empty() ? "senseless" : type;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/// common to InsetFloat and InsetWrap
|
|
|
|
docstring InsetCaptionable::floatName(string const & type) const
|
|
|
|
{
|
|
|
|
BufferParams const & bp = buffer().params();
|
|
|
|
FloatList const & floats = bp.documentClass().floats();
|
|
|
|
FloatList::const_iterator it = floats[type];
|
2015-10-04 19:38:47 +01:00
|
|
|
return (it == floats.end()) ? from_utf8(type) : bp.B_(it->second.name());
|
2015-09-02 19:53:13 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2016-09-24 00:49:00 +02:00
|
|
|
InsetCaption const * InsetCaptionable::getCaptionInset() const
|
|
|
|
{
|
|
|
|
ParagraphList::const_iterator pit = paragraphs().begin();
|
|
|
|
for (; pit != paragraphs().end(); ++pit) {
|
|
|
|
InsetList::const_iterator it = pit->insetList().begin();
|
|
|
|
for (; it != pit->insetList().end(); ++it) {
|
|
|
|
Inset & inset = *it->inset;
|
|
|
|
if (inset.lyxCode() == CAPTION_CODE) {
|
|
|
|
InsetCaption const * ins =
|
|
|
|
static_cast<InsetCaption const *>(it->inset);
|
|
|
|
return ins;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2020-11-15 03:03:16 +01:00
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
InsetLabel const * InsetCaptionable::getLabelInset() const
|
|
|
|
{
|
|
|
|
// A wrong hypothesis would be to limit the search to the caption: it is most likely there, but not necessarily!
|
|
|
|
|
|
|
|
// Iterate through the contents of the inset.
|
|
|
|
auto const end = paragraphs().end();
|
|
|
|
for (auto par = paragraphs().begin(); par != end; ++par) {
|
|
|
|
for (pos_type pos = 0; pos < par->size(); ++pos) {
|
|
|
|
const Inset * inset = par->getInset(pos);
|
|
|
|
|
|
|
|
// If this inset is a subfigure, skip it. Otherwise, you would return the label for the subfigure.
|
|
|
|
if (dynamic_cast<const InsetBox *>(inset)) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Maybe an inset is directly a label, in which case no more work is needed.
|
|
|
|
if (inset && dynamic_cast<const InsetLabel *>(inset))
|
|
|
|
return dynamic_cast<const InsetLabel *>(inset);
|
|
|
|
|
|
|
|
// More likely, the label is hidden in an inset of a paragraph (only if a subtype of InsetText). Thus,
|
|
|
|
// dig into that text.
|
|
|
|
if (!dynamic_cast<const InsetText *>(inset))
|
|
|
|
continue;
|
|
|
|
|
|
|
|
auto insetAsText = dynamic_cast<const InsetText *>(inset);
|
|
|
|
auto itIn = insetAsText->paragraphs().begin();
|
|
|
|
auto endIn = insetAsText->paragraphs().end();
|
|
|
|
for (; itIn != endIn; ++itIn) {
|
|
|
|
for (pos_type posIn = 0; posIn < itIn->size(); ++posIn) {
|
|
|
|
const Inset *insetIn = itIn->getInset(posIn);
|
|
|
|
if (insetIn && dynamic_cast<const InsetLabel *>(insetIn)) {
|
|
|
|
return dynamic_cast<const InsetLabel *>(insetIn);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Obviously, this solution does not scale with more levels of paragraphs and insets, but this should
|
|
|
|
// be enough: it is only used in captions.
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return nullptr;
|
2016-09-24 00:49:00 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
docstring InsetCaptionable::getCaptionText(OutputParams const & runparams) const
|
|
|
|
{
|
|
|
|
InsetCaption const * ins = getCaptionInset();
|
2020-11-15 03:03:16 +01:00
|
|
|
if (!ins)
|
2016-09-24 00:49:00 +02:00
|
|
|
return docstring();
|
|
|
|
|
|
|
|
odocstringstream ods;
|
|
|
|
ins->getCaptionAsPlaintext(ods, runparams);
|
|
|
|
return ods.str();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2020-06-08 23:27:49 +02:00
|
|
|
docstring InsetCaptionable::getCaptionDocBook(OutputParams const & runparams) const
|
|
|
|
{
|
|
|
|
InsetCaption const * ins = getCaptionInset();
|
|
|
|
if (ins == nullptr)
|
|
|
|
return docstring();
|
|
|
|
|
|
|
|
odocstringstream ods;
|
|
|
|
XMLStream xs(ods);
|
|
|
|
ins->getCaptionAsDocBook(xs, runparams);
|
|
|
|
return ods.str();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2016-09-24 00:49:00 +02:00
|
|
|
docstring InsetCaptionable::getCaptionHTML(OutputParams const & runparams) const
|
|
|
|
{
|
|
|
|
InsetCaption const * ins = getCaptionInset();
|
|
|
|
if (ins == 0)
|
|
|
|
return docstring();
|
|
|
|
|
|
|
|
odocstringstream ods;
|
2019-05-10 01:35:40 +02:00
|
|
|
XMLStream xs(ods);
|
2016-09-24 00:49:00 +02:00
|
|
|
docstring def = ins->getCaptionAsHTML(xs, runparams);
|
|
|
|
if (!def.empty())
|
|
|
|
// should already have been escaped
|
2019-05-10 01:35:40 +02:00
|
|
|
xs << XMLStream::ESCAPE_NONE << def << '\n';
|
2016-09-24 00:49:00 +02:00
|
|
|
return ods.str();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2015-09-27 07:05:00 +01:00
|
|
|
void InsetCaptionable::addToToc(DocIterator const & cpit, bool output_active,
|
2017-01-13 11:06:48 +01:00
|
|
|
UpdateType utype, TocBackend & backend) const
|
2015-09-02 19:53:13 +01:00
|
|
|
{
|
|
|
|
DocIterator pit = cpit;
|
|
|
|
pit.push_back(CursorSlice(const_cast<InsetCaptionable &>(*this)));
|
|
|
|
docstring str;
|
2015-09-27 07:05:00 +01:00
|
|
|
// Leave str empty if we generate for output (e.g. xhtml lists of figures).
|
|
|
|
// This ensures that there is a caption if and only if the string is
|
|
|
|
// non-empty.
|
|
|
|
if (utype != OutputUpdate)
|
|
|
|
text().forOutliner(str, TOC_ENTRY_LENGTH);
|
2017-01-13 11:06:48 +01:00
|
|
|
TocBuilder & b = backend.builder(caption_type_);
|
2016-06-02 21:40:11 +01:00
|
|
|
b.pushItem(pit, str, output_active);
|
2015-09-02 19:53:13 +01:00
|
|
|
// Proceed with the rest of the inset.
|
2017-10-16 10:12:21 +02:00
|
|
|
InsetCollapsible::addToToc(cpit, output_active, utype, backend);
|
2016-06-02 21:40:11 +01:00
|
|
|
b.pop();
|
2015-09-02 19:53:13 +01:00
|
|
|
}
|
|
|
|
|
2020-03-06 13:23:01 +01:00
|
|
|
void InsetCaptionable::updateBuffer(ParIterator const & it, UpdateType utype, bool const deleted)
|
2015-09-02 19:53:13 +01:00
|
|
|
{
|
|
|
|
Counters & cnts =
|
|
|
|
buffer().masterBuffer()->params().documentClass().counters();
|
|
|
|
string const saveflt = cnts.current_float();
|
|
|
|
bool const savesubflt = cnts.isSubfloat();
|
|
|
|
if (utype == OutputUpdate) {
|
|
|
|
// counters are local to the float
|
|
|
|
cnts.saveLastCounter();
|
|
|
|
}
|
|
|
|
bool const subflt = hasSubCaptions(it);
|
|
|
|
// floats can only embed subfloats of their own kind
|
|
|
|
if (subflt && !saveflt.empty() && saveflt != "senseless")
|
|
|
|
setCaptionType(saveflt);
|
|
|
|
// Tell captions what the current float is
|
|
|
|
cnts.current_float(caption_type_);
|
|
|
|
cnts.isSubfloat(subflt);
|
2020-03-06 13:23:01 +01:00
|
|
|
InsetCollapsible::updateBuffer(it, utype, deleted);
|
2015-09-02 19:53:13 +01:00
|
|
|
// Restore counters
|
|
|
|
cnts.current_float(saveflt);
|
|
|
|
if (utype == OutputUpdate)
|
|
|
|
cnts.restoreLastCounter();
|
|
|
|
cnts.isSubfloat(savesubflt);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
bool InsetCaptionable::insetAllowed(InsetCode c) const
|
|
|
|
{
|
2017-10-16 10:12:21 +02:00
|
|
|
return (c == CAPTION_CODE) || InsetCollapsible::insetAllowed(c);
|
2015-09-02 19:53:13 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
} // namespace lyx
|