mirror of
https://git.lyx.org/repos/lyx.git
synced 2025-01-11 11:08:41 +00:00
9e9c96035c
that also makes sure it doesn't do more work than it needs to do, by limiting the size to 40 characters. Previously, InsetBranch::addToToc() would have added a string representing the entire contents of the branch! It's hard to imagine that having to recalculate that sort of thing doesn't cause some problems with speed, especially in documents with lots of notes and branches and such. git-svn-id: svn://svn.lyx.org/lyx/lyx-devel/trunk@36974 a592a061-630c-0410-9148-cb99ea01b6c8
247 lines
5.1 KiB
C++
247 lines
5.1 KiB
C++
/**
|
|
* \file TocBackend.cpp
|
|
* This file is part of LyX, the document processor.
|
|
* Licence details can be found in the file COPYING.
|
|
*
|
|
* \author Jean-Marc Lasgouttes
|
|
* \author Angus Leeming
|
|
* \author Abdelrazak Younes
|
|
*
|
|
* Full author contact details are available in file CREDITS.
|
|
*/
|
|
|
|
#include <config.h>
|
|
|
|
#include "TocBackend.h"
|
|
|
|
#include "Buffer.h"
|
|
#include "BufferParams.h"
|
|
#include "FloatList.h"
|
|
#include "FuncRequest.h"
|
|
#include "InsetList.h"
|
|
#include "Layout.h"
|
|
#include "LyXAction.h"
|
|
#include "Paragraph.h"
|
|
#include "ParIterator.h"
|
|
#include "TextClass.h"
|
|
|
|
#include "insets/InsetArgument.h"
|
|
|
|
#include "support/convert.h"
|
|
#include "support/debug.h"
|
|
#include "support/docstream.h"
|
|
|
|
#include "support/lassert.h"
|
|
|
|
using namespace std;
|
|
|
|
|
|
namespace lyx {
|
|
|
|
///////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// TocItem implementation
|
|
//
|
|
///////////////////////////////////////////////////////////////////////////
|
|
|
|
TocItem::TocItem(DocIterator const & dit, int d, docstring const & s,
|
|
docstring const & t) : dit_(dit), depth_(d), str_(s), tooltip_(t)
|
|
{
|
|
}
|
|
|
|
|
|
int TocItem::id() const
|
|
{
|
|
return dit_.paragraph().id();
|
|
}
|
|
|
|
|
|
int TocItem::depth() const
|
|
{
|
|
return depth_;
|
|
}
|
|
|
|
|
|
docstring const & TocItem::str() const
|
|
{
|
|
return str_;
|
|
}
|
|
|
|
|
|
docstring const & TocItem::tooltip() const
|
|
{
|
|
return tooltip_;
|
|
}
|
|
|
|
|
|
docstring const TocItem::asString() const
|
|
{
|
|
return docstring(4 * depth_, ' ') + str_;
|
|
}
|
|
|
|
|
|
DocIterator const & TocItem::dit() const
|
|
{
|
|
return dit_;
|
|
}
|
|
|
|
|
|
FuncRequest TocItem::action() const
|
|
{
|
|
string const arg = convert<string>(dit_.paragraph().id())
|
|
+ ' ' + convert<string>(dit_.pos());
|
|
return FuncRequest(LFUN_PARAGRAPH_GOTO, arg);
|
|
}
|
|
|
|
|
|
///////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// TocBackend implementation
|
|
//
|
|
///////////////////////////////////////////////////////////////////////////
|
|
|
|
Toc const & TocBackend::toc(string const & type) const
|
|
{
|
|
// Is the type already supported?
|
|
TocList::const_iterator it = tocs_.find(type);
|
|
LASSERT(it != tocs_.end(), /**/);
|
|
|
|
return it->second;
|
|
}
|
|
|
|
|
|
Toc & TocBackend::toc(string const & type)
|
|
{
|
|
return tocs_[type];
|
|
}
|
|
|
|
|
|
bool TocBackend::updateItem(DocIterator const & dit)
|
|
{
|
|
if (dit.paragraph().layout().toclevel == Layout::NOT_IN_TOC)
|
|
return false;
|
|
|
|
if (toc("tableofcontents").empty()) {
|
|
// FIXME: should not happen,
|
|
// a call to TocBackend::update() is missing somewhere
|
|
LYXERR0("TocBackend::updateItem called but the TOC is empty!");
|
|
return false;
|
|
}
|
|
|
|
BufferParams const & bufparams = buffer_->params();
|
|
const int min_toclevel = bufparams.documentClass().min_toclevel();
|
|
|
|
TocIterator toc_item = item("tableofcontents", dit);
|
|
|
|
docstring tocstring;
|
|
|
|
// For each paragraph, traverse its insets and let them add
|
|
// their toc items
|
|
Paragraph & par = toc_item->dit_.paragraph();
|
|
InsetList::const_iterator it = par.insetList().begin();
|
|
InsetList::const_iterator end = par.insetList().end();
|
|
for (; it != end; ++it) {
|
|
Inset & inset = *it->inset;
|
|
if (inset.lyxCode() == ARG_CODE) {
|
|
if (!tocstring.empty())
|
|
break;
|
|
Paragraph const & inset_par =
|
|
*static_cast<InsetArgument&>(inset).paragraphs().begin();
|
|
if (!par.labelString().empty())
|
|
tocstring = par.labelString() + ' ';
|
|
tocstring += inset_par.asString(AS_STR_INSETS);
|
|
break;
|
|
}
|
|
}
|
|
|
|
int const toclevel = par.layout().toclevel;
|
|
if (toclevel != Layout::NOT_IN_TOC && toclevel >= min_toclevel
|
|
&& tocstring.empty())
|
|
tocstring = par.asString(AS_STR_LABEL | AS_STR_INSETS);
|
|
|
|
const_cast<TocItem &>(*toc_item).str_ = tocstring;
|
|
|
|
buffer_->updateTocItem("tableofcontents", dit);
|
|
return true;
|
|
}
|
|
|
|
|
|
void TocBackend::update()
|
|
{
|
|
tocs_.clear();
|
|
if (!buffer_->isInternal()) {
|
|
DocIterator dit;
|
|
buffer_->inset().addToToc(dit);
|
|
}
|
|
}
|
|
|
|
|
|
TocIterator TocBackend::item(string const & type,
|
|
DocIterator const & dit) const
|
|
{
|
|
TocList::const_iterator toclist_it = tocs_.find(type);
|
|
// Is the type supported?
|
|
LASSERT(toclist_it != tocs_.end(), /**/);
|
|
return toclist_it->second.item(dit);
|
|
}
|
|
|
|
|
|
TocIterator Toc::item(DocIterator const & dit) const
|
|
{
|
|
TocIterator last = begin();
|
|
TocIterator it = end();
|
|
if (it == last)
|
|
return it;
|
|
|
|
--it;
|
|
|
|
DocIterator dit_text = dit;
|
|
if (dit_text.inMathed()) {
|
|
// We are only interested in text so remove the math CursorSlice.
|
|
while (dit_text.inMathed())
|
|
dit_text.pop_back();
|
|
}
|
|
|
|
for (; it != last; --it) {
|
|
// We verify that we don't compare contents of two
|
|
// different document. This happens when you
|
|
// have parent and child documents.
|
|
if (&it->dit_[0].inset() != &dit_text[0].inset())
|
|
continue;
|
|
if (it->dit_ <= dit_text)
|
|
return it;
|
|
}
|
|
|
|
// We are before the first Toc Item:
|
|
return last;
|
|
}
|
|
|
|
|
|
Toc::iterator Toc::item(int depth, docstring const & str)
|
|
{
|
|
if (empty())
|
|
return end();
|
|
iterator it = begin();
|
|
iterator itend = end();
|
|
for (; it != itend; ++it) {
|
|
if (it->depth() == depth && it->str() == str)
|
|
break;
|
|
}
|
|
return it;
|
|
}
|
|
|
|
|
|
void TocBackend::writePlaintextTocList(string const & type, odocstream & os) const
|
|
{
|
|
TocList::const_iterator cit = tocs_.find(type);
|
|
if (cit != tocs_.end()) {
|
|
TocIterator ccit = cit->second.begin();
|
|
TocIterator end = cit->second.end();
|
|
for (; ccit != end; ++ccit)
|
|
os << ccit->asString() << from_utf8("\n");
|
|
}
|
|
}
|
|
|
|
|
|
} // namespace lyx
|