2018-12-18 13:08:55 -05:00
|
|
|
#include <qpdf/QPDFNumberTreeObjectHelper.hh>
|
2022-02-04 16:31:31 -05:00
|
|
|
|
2021-01-16 08:31:56 -05:00
|
|
|
#include <qpdf/NNTree.hh>
|
2021-11-04 13:52:47 -04:00
|
|
|
#include <qpdf/QIntC.hh>
|
2018-12-18 13:08:55 -05:00
|
|
|
|
2022-04-16 13:21:57 -04:00
|
|
|
namespace
|
2018-12-18 13:08:55 -05:00
|
|
|
{
|
2022-04-16 13:21:57 -04:00
|
|
|
class NumberTreeDetails: public NNTreeDetails
|
2018-12-18 13:08:55 -05:00
|
|
|
{
|
2022-04-16 13:21:57 -04:00
|
|
|
public:
|
|
|
|
virtual std::string const&
|
|
|
|
itemsKey() const override
|
|
|
|
{
|
|
|
|
static std::string k("/Nums");
|
|
|
|
return k;
|
|
|
|
}
|
|
|
|
virtual bool
|
|
|
|
keyValid(QPDFObjectHandle oh) const override
|
|
|
|
{
|
|
|
|
return oh.isInteger();
|
|
|
|
}
|
|
|
|
virtual 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
|
|
|
}
|
2022-04-16 13:21:57 -04:00
|
|
|
};
|
|
|
|
} // namespace
|
2021-01-16 08:31:56 -05:00
|
|
|
|
|
|
|
static NumberTreeDetails number_tree_details;
|
|
|
|
|
2022-08-07 09:09:04 -04:00
|
|
|
QPDFNumberTreeObjectHelper::~QPDFNumberTreeObjectHelper()
|
|
|
|
{
|
|
|
|
// Must be explicit and not inline -- see QPDF_DLL_CLASS in
|
|
|
|
// README-maintainer. For this specific class, see github issue
|
|
|
|
// #745.
|
|
|
|
}
|
|
|
|
|
2021-01-16 18:35:30 -05:00
|
|
|
QPDFNumberTreeObjectHelper::Members::Members(
|
2022-04-16 13:12:49 -04:00
|
|
|
QPDFObjectHandle& oh, QPDF& q, bool auto_repair) :
|
2021-01-16 18:35:30 -05:00
|
|
|
impl(std::make_shared<NNTreeImpl>(number_tree_details, q, oh, auto_repair))
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
QPDFNumberTreeObjectHelper::QPDFNumberTreeObjectHelper(
|
|
|
|
QPDFObjectHandle oh, QPDF& q, bool auto_repair) :
|
|
|
|
QPDFObjectHelper(oh),
|
2022-04-16 13:12:49 -04:00
|
|
|
m(new Members(oh, q, auto_repair))
|
2021-01-16 08:31:56 -05:00
|
|
|
{
|
|
|
|
}
|
2018-12-18 13:08:55 -05:00
|
|
|
|
2021-01-24 03:55:18 -05:00
|
|
|
QPDFNumberTreeObjectHelper
|
|
|
|
QPDFNumberTreeObjectHelper::newEmpty(QPDF& qpdf, bool auto_repair)
|
|
|
|
{
|
|
|
|
return QPDFNumberTreeObjectHelper(
|
2022-02-05 09:18:58 -05:00
|
|
|
qpdf.makeIndirectObject("<< /Nums [] >>"_qpdf), qpdf, auto_repair);
|
2021-01-24 03:55:18 -05:00
|
|
|
}
|
|
|
|
|
2021-01-16 12:11:17 -05: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);
|
2021-01-25 08:05:43 -05:00
|
|
|
updateIValue();
|
2021-01-16 12:11:17 -05:00
|
|
|
return *this;
|
|
|
|
}
|
|
|
|
|
|
|
|
QPDFNumberTreeObjectHelper::iterator&
|
|
|
|
QPDFNumberTreeObjectHelper::iterator::operator--()
|
|
|
|
{
|
|
|
|
--(*impl);
|
2021-01-25 08:05:43 -05:00
|
|
|
updateIValue();
|
2021-01-16 12:11:17 -05:00
|
|
|
return *this;
|
|
|
|
}
|
|
|
|
|
2021-01-25 08:05:43 -05:00
|
|
|
void
|
|
|
|
QPDFNumberTreeObjectHelper::iterator::updateIValue()
|
|
|
|
{
|
2022-04-02 17:14:10 -04:00
|
|
|
if (impl->valid()) {
|
2021-01-25 08:05:43 -05:00
|
|
|
auto p = *impl;
|
|
|
|
this->ivalue.first = p->first.getIntValue();
|
|
|
|
this->ivalue.second = p->second;
|
2022-04-02 17:14:10 -04:00
|
|
|
} else {
|
2021-01-25 08:05:43 -05:00
|
|
|
this->ivalue.first = 0;
|
|
|
|
this->ivalue.second = QPDFObjectHandle();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-01-16 12:11:17 -05:00
|
|
|
QPDFNumberTreeObjectHelper::iterator::reference
|
|
|
|
QPDFNumberTreeObjectHelper::iterator::operator*()
|
|
|
|
{
|
2021-01-25 08:05:43 -05:00
|
|
|
updateIValue();
|
|
|
|
return this->ivalue;
|
|
|
|
}
|
|
|
|
|
|
|
|
QPDFNumberTreeObjectHelper::iterator::pointer
|
|
|
|
QPDFNumberTreeObjectHelper::iterator::operator->()
|
|
|
|
{
|
|
|
|
updateIValue();
|
|
|
|
return &this->ivalue;
|
2021-01-16 12:11:17 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
bool
|
|
|
|
QPDFNumberTreeObjectHelper::iterator::operator==(iterator const& other) const
|
|
|
|
{
|
|
|
|
return *(impl) == *(other.impl);
|
|
|
|
}
|
|
|
|
|
2021-01-24 04:16:48 -05:00
|
|
|
void
|
|
|
|
QPDFNumberTreeObjectHelper::iterator::insertAfter(
|
|
|
|
numtree_number key, QPDFObjectHandle value)
|
|
|
|
{
|
|
|
|
impl->insertAfter(QPDFObjectHandle::newInteger(key), value);
|
2021-01-25 08:05:43 -05:00
|
|
|
updateIValue();
|
2021-01-24 04:16:48 -05:00
|
|
|
}
|
|
|
|
|
2021-01-24 11:48:46 -05:00
|
|
|
void
|
|
|
|
QPDFNumberTreeObjectHelper::iterator::remove()
|
|
|
|
{
|
|
|
|
impl->remove();
|
2021-01-25 08:05:43 -05:00
|
|
|
updateIValue();
|
2021-01-24 11:48:46 -05:00
|
|
|
}
|
|
|
|
|
2021-01-16 12:11:17 -05:00
|
|
|
QPDFNumberTreeObjectHelper::iterator
|
|
|
|
QPDFNumberTreeObjectHelper::begin() const
|
|
|
|
{
|
|
|
|
return iterator(std::make_shared<NNTreeIterator>(this->m->impl->begin()));
|
|
|
|
}
|
|
|
|
|
|
|
|
QPDFNumberTreeObjectHelper::iterator
|
|
|
|
QPDFNumberTreeObjectHelper::end() const
|
|
|
|
{
|
|
|
|
return iterator(std::make_shared<NNTreeIterator>(this->m->impl->end()));
|
|
|
|
}
|
|
|
|
|
|
|
|
QPDFNumberTreeObjectHelper::iterator
|
|
|
|
QPDFNumberTreeObjectHelper::last() const
|
|
|
|
{
|
|
|
|
return iterator(std::make_shared<NNTreeIterator>(this->m->impl->last()));
|
|
|
|
}
|
|
|
|
|
|
|
|
QPDFNumberTreeObjectHelper::iterator
|
2022-04-02 17:14:10 -04:00
|
|
|
QPDFNumberTreeObjectHelper::find(
|
|
|
|
numtree_number key, bool return_prev_if_not_found)
|
2021-01-16 12:11:17 -05:00
|
|
|
{
|
2022-04-02 17:14:10 -04:00
|
|
|
auto i = this->m->impl->find(
|
|
|
|
QPDFObjectHandle::newInteger(key), return_prev_if_not_found);
|
2021-01-16 12:11:17 -05:00
|
|
|
return iterator(std::make_shared<NNTreeIterator>(i));
|
|
|
|
}
|
|
|
|
|
2021-01-23 18:33:55 -05:00
|
|
|
QPDFNumberTreeObjectHelper::iterator
|
|
|
|
QPDFNumberTreeObjectHelper::insert(numtree_number key, QPDFObjectHandle value)
|
|
|
|
{
|
2022-04-02 17:14:10 -04:00
|
|
|
auto i = this->m->impl->insert(QPDFObjectHandle::newInteger(key), value);
|
2021-01-23 18:33:55 -05:00
|
|
|
return iterator(std::make_shared<NNTreeIterator>(i));
|
|
|
|
}
|
|
|
|
|
2021-01-24 11:48:46 -05:00
|
|
|
bool
|
2022-04-02 17:14:10 -04:00
|
|
|
QPDFNumberTreeObjectHelper::remove(numtree_number key, QPDFObjectHandle* value)
|
2021-01-24 11:48:46 -05:00
|
|
|
{
|
2022-04-02 17:14:10 -04:00
|
|
|
return this->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()
|
|
|
|
{
|
2021-01-16 12:11:17 -05:00
|
|
|
auto i = begin();
|
2022-04-02 17:14:10 -04:00
|
|
|
if (i == end()) {
|
2018-12-18 13:08:55 -05:00
|
|
|
return 0;
|
|
|
|
}
|
2021-01-25 08:05:43 -05:00
|
|
|
return i->first;
|
2018-12-18 13:08:55 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
QPDFNumberTreeObjectHelper::numtree_number
|
|
|
|
QPDFNumberTreeObjectHelper::getMax()
|
|
|
|
{
|
2021-01-16 12:11:17 -05:00
|
|
|
auto i = last();
|
2022-04-02 17:14:10 -04:00
|
|
|
if (i == end()) {
|
2018-12-18 13:08:55 -05:00
|
|
|
return 0;
|
|
|
|
}
|
2021-01-25 08:05:43 -05:00
|
|
|
return i->first;
|
2018-12-18 13:08:55 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
bool
|
|
|
|
QPDFNumberTreeObjectHelper::hasIndex(numtree_number idx)
|
|
|
|
{
|
2021-01-16 12:11:17 -05:00
|
|
|
auto i = find(idx);
|
|
|
|
return (i != this->end());
|
2018-12-18 13:08:55 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
bool
|
2022-04-02 17:14:10 -04:00
|
|
|
QPDFNumberTreeObjectHelper::findObject(numtree_number idx, QPDFObjectHandle& oh)
|
2018-12-18 13:08:55 -05:00
|
|
|
{
|
2021-01-16 12:11:17 -05:00
|
|
|
auto i = find(idx);
|
2022-04-02 17:14:10 -04:00
|
|
|
if (i == end()) {
|
2018-12-18 13:08:55 -05:00
|
|
|
return false;
|
|
|
|
}
|
2021-01-25 08:05:43 -05:00
|
|
|
oh = i->second;
|
2018-12-18 13:08:55 -05:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool
|
|
|
|
QPDFNumberTreeObjectHelper::findObjectAtOrBelow(
|
2022-04-02 17:14:10 -04:00
|
|
|
numtree_number idx, QPDFObjectHandle& oh, numtree_number& offset)
|
2018-12-18 13:08:55 -05:00
|
|
|
{
|
2021-01-16 12:11:17 -05:00
|
|
|
auto i = find(idx, true);
|
2022-04-02 17:14:10 -04:00
|
|
|
if (i == end()) {
|
2018-12-18 13:08:55 -05:00
|
|
|
return false;
|
|
|
|
}
|
2021-01-25 08:05:43 -05:00
|
|
|
oh = i->second;
|
2021-11-04 13:52:47 -04:00
|
|
|
QIntC::range_check_substract(idx, i->first);
|
2021-01-25 08:05:43 -05:00
|
|
|
offset = idx - i->first;
|
2018-12-18 13:08:55 -05:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2021-01-23 18:33:55 -05:00
|
|
|
void
|
|
|
|
QPDFNumberTreeObjectHelper::setSplitThreshold(int t)
|
|
|
|
{
|
|
|
|
this->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;
|
2021-01-16 12:11:17 -05:00
|
|
|
result.insert(begin(), end());
|
2018-12-18 13:08:55 -05:00
|
|
|
return result;
|
|
|
|
}
|