2
1
mirror of https://github.com/qpdf/qpdf.git synced 2025-01-22 22:58:33 +00:00
qpdf/libqpdf/QPDFNumberTreeObjectHelper.cc

239 lines
5.5 KiB
C++
Raw Normal View History

2018-12-18 13:08:55 -05:00
#include <qpdf/QPDFNumberTreeObjectHelper.hh>
#include <qpdf/NNTree.hh>
#include <qpdf/QIntC.hh>
2018-12-18 13:08:55 -05:00
namespace
2018-12-18 13:08:55 -05:00
{
class NumberTreeDetails: public NNTreeDetails
2018-12-18 13:08:55 -05:00
{
public:
2023-05-20 13:56:33 +01:00
std::string const&
itemsKey() const override
{
static std::string k("/Nums");
return k;
}
2023-05-20 13:56:33 +01:00
bool
keyValid(QPDFObjectHandle oh) const override
{
return oh.isInteger();
}
2023-05-20 13:56:33 +01:00
int
compareKeys(QPDFObjectHandle a, QPDFObjectHandle b) const override
{
if (!(keyValid(a) && keyValid(b))) {
// We don't call this without calling keyValid first
throw std::logic_error("comparing invalid keys");
}
auto as = a.getIntValue();
auto bs = b.getIntValue();
return ((as < bs) ? -1 : (as > bs) ? 1 : 0);
2018-12-18 13:08:55 -05:00
}
};
} // namespace
static NumberTreeDetails number_tree_details;
QPDFNumberTreeObjectHelper::~QPDFNumberTreeObjectHelper() // NOLINT (modernize-use-equals-default)
{
// Must be explicit and not inline -- see QPDF_DLL_CLASS in README-maintainer. For this specific
// class, see github issue #745.
}
2023-05-21 13:35:09 -04:00
QPDFNumberTreeObjectHelper::Members::Members(QPDFObjectHandle& oh, QPDF& q, bool auto_repair) :
impl(std::make_shared<NNTreeImpl>(number_tree_details, q, oh, auto_repair))
{
}
QPDFNumberTreeObjectHelper::QPDFNumberTreeObjectHelper(
QPDFObjectHandle oh, QPDF& q, bool auto_repair) :
QPDFObjectHelper(oh),
m(new Members(oh, q, auto_repair))
{
}
2018-12-18 13:08:55 -05:00
QPDFNumberTreeObjectHelper
QPDFNumberTreeObjectHelper::newEmpty(QPDF& qpdf, bool auto_repair)
{
return {qpdf.makeIndirectObject("<< /Nums [] >>"_qpdf), qpdf, auto_repair};
}
2023-05-21 13:35:09 -04:00
QPDFNumberTreeObjectHelper::iterator::iterator(std::shared_ptr<NNTreeIterator> const& i) :
impl(i)
{
}
bool
QPDFNumberTreeObjectHelper::iterator::valid() const
{
return impl->valid();
}
QPDFNumberTreeObjectHelper::iterator&
QPDFNumberTreeObjectHelper::iterator::operator++()
{
++(*impl);
updateIValue();
return *this;
}
QPDFNumberTreeObjectHelper::iterator&
QPDFNumberTreeObjectHelper::iterator::operator--()
{
--(*impl);
updateIValue();
return *this;
}
void
QPDFNumberTreeObjectHelper::iterator::updateIValue()
{
if (impl->valid()) {
auto p = *impl;
this->ivalue.first = p->first.getIntValue();
this->ivalue.second = p->second;
} else {
this->ivalue.first = 0;
this->ivalue.second = QPDFObjectHandle();
}
}
QPDFNumberTreeObjectHelper::iterator::reference
QPDFNumberTreeObjectHelper::iterator::operator*()
{
updateIValue();
return this->ivalue;
}
QPDFNumberTreeObjectHelper::iterator::pointer
QPDFNumberTreeObjectHelper::iterator::operator->()
{
updateIValue();
return &this->ivalue;
}
bool
QPDFNumberTreeObjectHelper::iterator::operator==(iterator const& other) const
{
return *(impl) == *(other.impl);
}
2021-01-24 04:16:48 -05:00
void
2023-05-21 13:35:09 -04:00
QPDFNumberTreeObjectHelper::iterator::insertAfter(numtree_number key, QPDFObjectHandle value)
2021-01-24 04:16:48 -05:00
{
impl->insertAfter(QPDFObjectHandle::newInteger(key), value);
updateIValue();
2021-01-24 04:16:48 -05:00
}
2021-01-24 11:48:46 -05:00
void
QPDFNumberTreeObjectHelper::iterator::remove()
{
impl->remove();
updateIValue();
2021-01-24 11:48:46 -05:00
}
QPDFNumberTreeObjectHelper::iterator
QPDFNumberTreeObjectHelper::begin() const
{
return {std::make_shared<NNTreeIterator>(m->impl->begin())};
}
QPDFNumberTreeObjectHelper::iterator
QPDFNumberTreeObjectHelper::end() const
{
return {std::make_shared<NNTreeIterator>(m->impl->end())};
}
QPDFNumberTreeObjectHelper::iterator
QPDFNumberTreeObjectHelper::last() const
{
return {std::make_shared<NNTreeIterator>(m->impl->last())};
}
QPDFNumberTreeObjectHelper::iterator
2023-05-21 13:35:09 -04:00
QPDFNumberTreeObjectHelper::find(numtree_number key, bool return_prev_if_not_found)
{
2023-05-21 13:35:09 -04:00
auto i = m->impl->find(QPDFObjectHandle::newInteger(key), return_prev_if_not_found);
return {std::make_shared<NNTreeIterator>(i)};
}
QPDFNumberTreeObjectHelper::iterator
QPDFNumberTreeObjectHelper::insert(numtree_number key, QPDFObjectHandle value)
{
auto i = m->impl->insert(QPDFObjectHandle::newInteger(key), value);
return {std::make_shared<NNTreeIterator>(i)};
}
2021-01-24 11:48:46 -05:00
bool
QPDFNumberTreeObjectHelper::remove(numtree_number key, QPDFObjectHandle* value)
2021-01-24 11:48:46 -05:00
{
return m->impl->remove(QPDFObjectHandle::newInteger(key), value);
2021-01-24 11:48:46 -05:00
}
2018-12-18 13:08:55 -05:00
QPDFNumberTreeObjectHelper::numtree_number
QPDFNumberTreeObjectHelper::getMin()
{
auto i = begin();
if (i == end()) {
2018-12-18 13:08:55 -05:00
return 0;
}
return i->first;
2018-12-18 13:08:55 -05:00
}
QPDFNumberTreeObjectHelper::numtree_number
QPDFNumberTreeObjectHelper::getMax()
{
auto i = last();
if (i == end()) {
2018-12-18 13:08:55 -05:00
return 0;
}
return i->first;
2018-12-18 13:08:55 -05:00
}
bool
QPDFNumberTreeObjectHelper::hasIndex(numtree_number idx)
{
auto i = find(idx);
return (i != this->end());
2018-12-18 13:08:55 -05:00
}
bool
QPDFNumberTreeObjectHelper::findObject(numtree_number idx, QPDFObjectHandle& oh)
2018-12-18 13:08:55 -05:00
{
auto i = find(idx);
if (i == end()) {
2018-12-18 13:08:55 -05:00
return false;
}
oh = i->second;
2018-12-18 13:08:55 -05:00
return true;
}
bool
QPDFNumberTreeObjectHelper::findObjectAtOrBelow(
numtree_number idx, QPDFObjectHandle& oh, numtree_number& offset)
2018-12-18 13:08:55 -05:00
{
auto i = find(idx, true);
if (i == end()) {
2018-12-18 13:08:55 -05:00
return false;
}
oh = i->second;
QIntC::range_check_substract(idx, i->first);
offset = idx - i->first;
2018-12-18 13:08:55 -05:00
return true;
}
void
QPDFNumberTreeObjectHelper::setSplitThreshold(int t)
{
m->impl->setSplitThreshold(t);
}
2018-12-18 13:08:55 -05:00
std::map<QPDFNumberTreeObjectHelper::numtree_number, QPDFObjectHandle>
QPDFNumberTreeObjectHelper::getAsMap() const
{
std::map<numtree_number, QPDFObjectHandle> result;
result.insert(begin(), end());
2018-12-18 13:08:55 -05:00
return result;
}