7#include "core/fpdfapi/page/cpdf_function.h"
13#include "core/fpdfapi/page/cpdf_expintfunc.h"
14#include "core/fpdfapi/page/cpdf_psfunc.h"
15#include "core/fpdfapi/page/cpdf_sampledfunc.h"
16#include "core/fpdfapi/page/cpdf_stitchfunc.h"
17#include "core/fpdfapi/parser/cpdf_array.h"
18#include "core/fpdfapi/parser/cpdf_dictionary.h"
19#include "core/fpdfapi/parser/cpdf_stream.h"
20#include "core/fpdfapi/parser/fpdf_parser_utility.h"
21#include "core/fxcrt/containers/contains.h"
22#include "core/fxcrt/fx_safe_types.h"
23#include "core/fxcrt/scoped_set_insertion.h"
24#include "core/fxcrt/stl_util.h"
46 return Load(std::move(pFuncObj), &visited);
52 VisitedSet* pVisited) {
56 if (
pdfium::Contains(*pVisited, pFuncObj))
59 ScopedSetInsertion<VisitedSet::value_type> insertion(pVisited, pFuncObj);
62 if (
const CPDF_Stream* pStream = pFuncObj->AsStream())
63 iType = pStream
->GetDict()->GetIntegerFor(
"FunctionType");
67 std::unique_ptr<CPDF_Function> pFunc;
68 Type type = IntegerToFunctionType(iType);
70 pFunc =
std::make_unique<CPDF_SampledFunc>();
72 pFunc =
std::make_unique<CPDF_ExpIntFunc>();
74 pFunc =
std::make_unique<CPDF_StitchFunc>();
76 pFunc =
std::make_unique<CPDF_PSFunc>();
78 if (!pFunc || !pFunc->Init(pFuncObj, pVisited))
91 pStream ? pStream->GetDict() : pdfium::WrapRetain(pObj->AsDictionary());
102 m_Domains = ReadArrayElementsToVector(pDomains.Get(), nInputs);
105 m_nOutputs = pRanges ?
fxcrt::CollectionSize<uint32_t>(*pRanges) / 2 : 0;
109 bool bRangeRequired =
116 m_Ranges = ReadArrayElementsToVector(pRanges.Get(), nOutputs);
120 if (!v_Init(pObj, pVisited))
123 if (!m_Ranges.empty() && m_nOutputs > old_outputs) {
126 m_Ranges.resize(nOutputs.ValueOrDie());
132 pdfium::span<
float> results)
const {
136 std::vector<
float> clamped_inputs(
m_nInputs);
137 for (uint32_t i = 0; i <
m_nInputs; i++) {
138 float domain1 = m_Domains[i * 2];
139 float domain2 = m_Domains[i * 2 + 1];
140 if (domain1 > domain2)
143 clamped_inputs[i] =
std::clamp(inputs[i], domain1, domain2);
145 if (!v_Call(clamped_inputs, results))
148 if (m_Ranges.empty())
152 float range1 = m_Ranges[i * 2];
153 float range2 = m_Ranges[i * 2 + 1];
157 results[i] =
std::clamp(results[i], range1, range2);
168 float divisor = xmax - xmin;
169 return ymin + (divisor ? (x - xmin) * (ymax - ymin) / divisor : 0);
172#if defined(PDF_USE_SKIA)
173const CPDF_SampledFunc* CPDF_Function::ToSampledFunc()
const {
174 return m_Type == Type::kType0Sampled
175 ?
static_cast<
const CPDF_SampledFunc*>(
this)
179const CPDF_ExpIntFunc* CPDF_Function::ToExpIntFunc()
const {
180 return m_Type == Type::kType2ExponentialInterpolation
181 ?
static_cast<
const CPDF_ExpIntFunc*>(
this)
185const CPDF_StitchFunc* CPDF_Function::ToStitchFunc()
const {
186 return m_Type == Type::kType3Stitching
187 ?
static_cast<
const CPDF_StitchFunc*>(
this)
std::vector< RetainPtr< CPDF_Object > >::const_iterator const_iterator
int GetIntegerFor(const ByteString &key) const
std::map< ByteString, RetainPtr< CPDF_Object >, std::less<> > DictMap
bool Init(const CPDF_Object *pObj, VisitedSet *pVisited)
std::optional< uint32_t > Call(pdfium::span< const float > inputs, pdfium::span< float > results) const
@ kType2ExponentialInterpolation
float Interpolate(float x, float xmin, float xmax, float ymin, float ymax) const
RetainPtr< const CPDF_Dictionary > GetDict() const
const CPDF_Stream * AsStream() const