| // Copyright 2016 The PDFium Authors |
| // Use of this source code is governed by a BSD-style license that can be |
| // found in the LICENSE file. |
| |
| #include "core/fpdfdoc/cpdf_formfield.h" |
| |
| #include <vector> |
| |
| #include "constants/form_fields.h" |
| #include "constants/form_flags.h" |
| #include "core/fpdfapi/page/cpdf_pagemodule.h" |
| #include "core/fpdfapi/parser/cpdf_array.h" |
| #include "core/fpdfapi/parser/cpdf_dictionary.h" |
| #include "core/fpdfapi/parser/cpdf_document.h" |
| #include "core/fpdfapi/parser/cpdf_indirect_object_holder.h" |
| #include "core/fpdfapi/parser/cpdf_name.h" |
| #include "core/fpdfapi/parser/cpdf_number.h" |
| #include "core/fpdfapi/parser/cpdf_reference.h" |
| #include "core/fpdfapi/parser/cpdf_string.h" |
| #include "core/fpdfapi/parser/cpdf_test_document.h" |
| #include "core/fpdfdoc/cpdf_interactiveform.h" |
| #include "core/fxcrt/fx_memory.h" |
| #include "testing/gtest/include/gtest/gtest.h" |
| #include "third_party/base/containers/contains.h" |
| |
| namespace { |
| |
| // Create and destroys the page module that is necessary when instantiating a |
| // CPDF_Document. |
| class ScopedCPDF_PageModule { |
| public: |
| FX_STACK_ALLOCATED(); |
| |
| ScopedCPDF_PageModule() { CPDF_PageModule::Create(); } |
| ~ScopedCPDF_PageModule() { CPDF_PageModule::Destroy(); } |
| }; |
| |
| void TestMultiselectFieldDict(RetainPtr<CPDF_Array> opt_array, |
| RetainPtr<CPDF_Object> values, |
| RetainPtr<CPDF_Object> selected_indices, |
| bool expected_use_indices, |
| const std::vector<int>& expected_indices, |
| const std::vector<int>& excluded_indices) { |
| auto form_dict = pdfium::MakeRetain<CPDF_Dictionary>(); |
| form_dict->SetNewFor<CPDF_Name>("Type", "Annot"); |
| form_dict->SetNewFor<CPDF_Name>("Subtype", "Widget"); |
| form_dict->SetNewFor<CPDF_Name>(pdfium::form_fields::kFT, |
| pdfium::form_fields::kCh); |
| constexpr int kMuliSelectFlag = pdfium::form_flags::kChoiceMultiSelect; |
| form_dict->SetNewFor<CPDF_Number>(pdfium::form_fields::kFf, kMuliSelectFlag); |
| form_dict->SetFor("Opt", opt_array); |
| form_dict->SetFor(pdfium::form_fields::kV, values); |
| form_dict->SetFor("I", selected_indices); |
| |
| CPDF_TestDocument doc; |
| CPDF_InteractiveForm form(&doc); |
| CPDF_FormField form_field(&form, std::move(form_dict)); |
| EXPECT_EQ(expected_use_indices, form_field.UseSelectedIndicesObject()); |
| for (int i = 0; i < form_field.CountOptions(); i++) { |
| const bool expected_selected = pdfium::Contains(expected_indices, i); |
| EXPECT_EQ(expected_selected, form_field.IsItemSelected(i)); |
| } |
| for (int i : excluded_indices) { |
| EXPECT_FALSE(form_field.IsItemSelected(i)); |
| } |
| } |
| |
| } // namespace |
| |
| TEST(CPDF_FormFieldTest, GetFullNameForDict) { |
| WideString name = CPDF_FormField::GetFullNameForDict(nullptr); |
| EXPECT_TRUE(name.IsEmpty()); |
| |
| CPDF_IndirectObjectHolder obj_holder; |
| auto root = obj_holder.NewIndirect<CPDF_Dictionary>(); |
| root->SetNewFor<CPDF_Name>("T", "foo"); |
| name = CPDF_FormField::GetFullNameForDict(root.Get()); |
| EXPECT_STREQ("foo", name.ToUTF8().c_str()); |
| |
| auto dict1 = obj_holder.NewIndirect<CPDF_Dictionary>(); |
| root->SetNewFor<CPDF_Reference>("Parent", &obj_holder, dict1->GetObjNum()); |
| dict1->SetNewFor<CPDF_Name>("T", "bar"); |
| name = CPDF_FormField::GetFullNameForDict(root.Get()); |
| EXPECT_STREQ("bar.foo", name.ToUTF8().c_str()); |
| |
| auto dict2 = dict1->SetNewFor<CPDF_Dictionary>("Parent"); |
| name = CPDF_FormField::GetFullNameForDict(root.Get()); |
| EXPECT_STREQ("bar.foo", name.ToUTF8().c_str()); |
| |
| auto dict3 = obj_holder.NewIndirect<CPDF_Dictionary>(); |
| dict2->SetNewFor<CPDF_Reference>("Parent", &obj_holder, dict3->GetObjNum()); |
| |
| dict3->SetNewFor<CPDF_Name>("T", "qux"); |
| name = CPDF_FormField::GetFullNameForDict(root.Get()); |
| EXPECT_STREQ("qux.bar.foo", name.ToUTF8().c_str()); |
| |
| dict3->SetNewFor<CPDF_Reference>("Parent", &obj_holder, root->GetObjNum()); |
| name = CPDF_FormField::GetFullNameForDict(root.Get()); |
| EXPECT_STREQ("qux.bar.foo", name.ToUTF8().c_str()); |
| name = CPDF_FormField::GetFullNameForDict(dict1.Get()); |
| EXPECT_STREQ("foo.qux.bar", name.ToUTF8().c_str()); |
| name = CPDF_FormField::GetFullNameForDict(dict2.Get()); |
| EXPECT_STREQ("bar.foo.qux", name.ToUTF8().c_str()); |
| name = CPDF_FormField::GetFullNameForDict(dict3.Get()); |
| EXPECT_STREQ("bar.foo.qux", name.ToUTF8().c_str()); |
| } |
| |
| TEST(CPDF_FormFieldTest, IsItemSelected) { |
| ScopedCPDF_PageModule page_module; |
| |
| auto opt_array = pdfium::MakeRetain<CPDF_Array>(); |
| opt_array->AppendNew<CPDF_String>(L"Alpha"); |
| opt_array->AppendNew<CPDF_String>(L"Beta"); |
| opt_array->AppendNew<CPDF_String>(L"Gamma"); |
| opt_array->AppendNew<CPDF_String>(L"Delta"); |
| opt_array->AppendNew<CPDF_String>(L"Epsilon"); |
| |
| { |
| // No Values (/V) or Selected Indices (/I) objects. |
| std::vector<int> expected_indices; |
| std::vector<int> excluded_indices{-1, 5}; |
| TestMultiselectFieldDict(opt_array, /*values=*/nullptr, |
| /*selected_indices=*/nullptr, |
| /*expected_use_indices=*/false, expected_indices, |
| excluded_indices); |
| } |
| { |
| // Values (/V) object is just a string. |
| auto values = pdfium::MakeRetain<CPDF_String>(/*pPool=*/nullptr, L"Gamma"); |
| std::vector<int> expected_indices{2}; |
| std::vector<int> excluded_indices{-1, 5}; |
| TestMultiselectFieldDict(opt_array, values, /*selected_indices=*/nullptr, |
| /*expected_use_indices=*/false, expected_indices, |
| excluded_indices); |
| } |
| { |
| // Values (/V) object is just an invalid string. |
| auto values = pdfium::MakeRetain<CPDF_String>(/*pPool=*/nullptr, L"Omega"); |
| std::vector<int> expected_indices; |
| std::vector<int> excluded_indices{-1, 5}; |
| TestMultiselectFieldDict(opt_array, values, /*selected_indices=*/nullptr, |
| /*expected_use_indices=*/false, expected_indices, |
| excluded_indices); |
| } |
| { |
| // Values (/V) object is an array with one object. |
| auto values = pdfium::MakeRetain<CPDF_Array>(); |
| values->AppendNew<CPDF_String>(L"Beta"); |
| std::vector<int> expected_indices{1}; |
| std::vector<int> excluded_indices{-1, 5}; |
| TestMultiselectFieldDict(opt_array, values, /*selected_indices=*/nullptr, |
| /*expected_use_indices=*/false, expected_indices, |
| excluded_indices); |
| } |
| { |
| // Values (/V) object is an array with one invalid object. |
| auto values = pdfium::MakeRetain<CPDF_Array>(); |
| values->AppendNew<CPDF_String>(L"Omega"); |
| std::vector<int> expected_indices; |
| std::vector<int> excluded_indices{-1, 5}; |
| TestMultiselectFieldDict(opt_array, values, /*selected_indices=*/nullptr, |
| /*expected_use_indices=*/false, expected_indices, |
| excluded_indices); |
| } |
| { |
| // Values (/V) object is an array with multiple objects. |
| auto values = pdfium::MakeRetain<CPDF_Array>(); |
| values->AppendNew<CPDF_String>(L"Beta"); |
| values->AppendNew<CPDF_String>(L"Epsilon"); |
| std::vector<int> expected_indices{1, 4}; |
| std::vector<int> excluded_indices{-1, 5}; |
| TestMultiselectFieldDict(opt_array, values, /*selected_indices=*/nullptr, |
| /*expected_use_indices=*/false, expected_indices, |
| excluded_indices); |
| } |
| { |
| // Values (/V) object is an array with multiple objects with one invalid. |
| auto values = pdfium::MakeRetain<CPDF_Array>(); |
| values->AppendNew<CPDF_String>(L"Beta"); |
| values->AppendNew<CPDF_String>(L"Epsilon"); |
| values->AppendNew<CPDF_String>(L"Omega"); |
| std::vector<int> expected_indices{1, 4}; |
| std::vector<int> excluded_indices{-1, 5}; |
| TestMultiselectFieldDict(opt_array, values, /*selected_indices=*/nullptr, |
| /*expected_use_indices=*/false, expected_indices, |
| excluded_indices); |
| } |
| { |
| // Selected indices (/I) object is just a number. |
| auto selected_indices = pdfium::MakeRetain<CPDF_Number>(3); |
| std::vector<int> expected_indices{3}; |
| std::vector<int> excluded_indices{-1, 5}; |
| TestMultiselectFieldDict(opt_array, /*values=*/nullptr, selected_indices, |
| /*expected_use_indices=*/true, expected_indices, |
| excluded_indices); |
| } |
| { |
| // Selected indices (/I) object is just an invalid number. |
| auto selected_indices = pdfium::MakeRetain<CPDF_Number>(26); |
| std::vector<int> expected_indices; |
| std::vector<int> excluded_indices{-1, 5, 26}; |
| TestMultiselectFieldDict(opt_array, /*values=*/nullptr, selected_indices, |
| /*expected_use_indices=*/true, expected_indices, |
| excluded_indices); |
| } |
| { |
| // Selected indices (/I) object is an array with one object. |
| auto selected_indices = pdfium::MakeRetain<CPDF_Array>(); |
| selected_indices->AppendNew<CPDF_Number>(0); |
| std::vector<int> expected_indices{0}; |
| std::vector<int> excluded_indices{-1, 5}; |
| TestMultiselectFieldDict(opt_array, /*values=*/nullptr, selected_indices, |
| /*expected_use_indices=*/true, expected_indices, |
| excluded_indices); |
| } |
| { |
| // Selected indices (/I) object is an array with multiple objects. |
| auto selected_indices = pdfium::MakeRetain<CPDF_Array>(); |
| selected_indices->AppendNew<CPDF_Number>(0); |
| selected_indices->AppendNew<CPDF_Number>(2); |
| selected_indices->AppendNew<CPDF_Number>(3); |
| std::vector<int> expected_indices{0, 2, 3}; |
| std::vector<int> excluded_indices{-1, 5}; |
| TestMultiselectFieldDict(opt_array, /*values=*/nullptr, selected_indices, |
| /*expected_use_indices=*/true, expected_indices, |
| excluded_indices); |
| } |
| { |
| // Selected indices (/I) object is an array with multiple objects and some |
| // are invalid. |
| auto selected_indices = pdfium::MakeRetain<CPDF_Array>(); |
| selected_indices->AppendNew<CPDF_Number>(0); |
| selected_indices->AppendNew<CPDF_Number>(2); |
| selected_indices->AppendNew<CPDF_Number>(3); |
| selected_indices->AppendNew<CPDF_Number>(-5); |
| selected_indices->AppendNew<CPDF_Number>(12); |
| selected_indices->AppendNew<CPDF_Number>(42); |
| std::vector<int> expected_indices{0, 2, 3}; |
| std::vector<int> excluded_indices{-5, -1, 5, 12, 42}; |
| TestMultiselectFieldDict(opt_array, /*values=*/nullptr, selected_indices, |
| /*expected_use_indices=*/true, expected_indices, |
| excluded_indices); |
| } |
| { |
| // Values (/V) or Selected Indices (/I) objects conflict with different |
| // lengths. |
| auto values = pdfium::MakeRetain<CPDF_Array>(); |
| values->AppendNew<CPDF_String>(L"Beta"); |
| values->AppendNew<CPDF_String>(L"Epsilon"); |
| auto selected_indices = pdfium::MakeRetain<CPDF_Array>(); |
| selected_indices->AppendNew<CPDF_Number>(0); |
| selected_indices->AppendNew<CPDF_Number>(2); |
| selected_indices->AppendNew<CPDF_Number>(3); |
| std::vector<int> expected_indices{1, 4}; |
| std::vector<int> excluded_indices{-1, 5}; |
| TestMultiselectFieldDict(opt_array, values, selected_indices, |
| /*expected_use_indices=*/false, expected_indices, |
| excluded_indices); |
| } |
| { |
| // Values (/V) or Selected Indices (/I) objects conflict with same lengths. |
| auto values = pdfium::MakeRetain<CPDF_Array>(); |
| values->AppendNew<CPDF_String>(L"Alpha"); |
| values->AppendNew<CPDF_String>(L"Epsilon"); |
| auto selected_indices = pdfium::MakeRetain<CPDF_Array>(); |
| selected_indices->AppendNew<CPDF_Number>(2); |
| selected_indices->AppendNew<CPDF_Number>(3); |
| std::vector<int> expected_indices{0, 4}; |
| std::vector<int> excluded_indices{-1, 5}; |
| TestMultiselectFieldDict(opt_array, values, selected_indices, |
| /*expected_use_indices=*/false, expected_indices, |
| excluded_indices); |
| } |
| { |
| // Values (/V) or Selected Indices (/I) objects conflict with values being |
| // invalid. |
| auto values = pdfium::MakeRetain<CPDF_Array>(); |
| values->AppendNew<CPDF_String>(L"Beta"); |
| values->AppendNew<CPDF_String>(L"Epsilon"); |
| values->AppendNew<CPDF_String>(L"Omega"); |
| auto selected_indices = pdfium::MakeRetain<CPDF_Array>(); |
| selected_indices->AppendNew<CPDF_Number>(1); |
| selected_indices->AppendNew<CPDF_Number>(4); |
| std::vector<int> expected_indices{1, 4}; |
| std::vector<int> excluded_indices{-1, 5}; |
| TestMultiselectFieldDict(opt_array, values, selected_indices, |
| /*expected_use_indices=*/false, expected_indices, |
| excluded_indices); |
| } |
| { |
| // Values (/V) or Selected Indices (/I) objects conflict with selected |
| // indices being invalid. |
| auto values = pdfium::MakeRetain<CPDF_Array>(); |
| values->AppendNew<CPDF_String>(L"Beta"); |
| values->AppendNew<CPDF_String>(L"Epsilon"); |
| auto selected_indices = pdfium::MakeRetain<CPDF_Array>(); |
| selected_indices->AppendNew<CPDF_Number>(1); |
| selected_indices->AppendNew<CPDF_Number>(4); |
| selected_indices->AppendNew<CPDF_Number>(26); |
| std::vector<int> expected_indices{1, 4}; |
| std::vector<int> excluded_indices{-1, 5, 26}; |
| TestMultiselectFieldDict(opt_array, values, selected_indices, |
| /*expected_use_indices=*/false, expected_indices, |
| excluded_indices); |
| } |
| { |
| // Values (/V) or Selected Indices (/I) objects conflict with both being |
| // invalid. |
| auto values = pdfium::MakeRetain<CPDF_Array>(); |
| values->AppendNew<CPDF_String>(L"Beta"); |
| values->AppendNew<CPDF_String>(L"Epsilon"); |
| values->AppendNew<CPDF_String>(L"Omega"); |
| auto selected_indices = pdfium::MakeRetain<CPDF_Array>(); |
| selected_indices->AppendNew<CPDF_Number>(0); |
| selected_indices->AppendNew<CPDF_Number>(2); |
| selected_indices->AppendNew<CPDF_Number>(3); |
| selected_indices->AppendNew<CPDF_Number>(26); |
| std::vector<int> expected_indices{1, 4}; |
| std::vector<int> excluded_indices{-1, 5, 26}; |
| TestMultiselectFieldDict(opt_array, values, selected_indices, |
| /*expected_use_indices=*/false, expected_indices, |
| excluded_indices); |
| } |
| { |
| // Values (/V) or Selected Indices (/I) objects conflict with each not being |
| // an array. |
| auto values = pdfium::MakeRetain<CPDF_String>(/*pPool=*/nullptr, L"Gamma"); |
| auto selected_indices = pdfium::MakeRetain<CPDF_Number>(4); |
| std::vector<int> expected_indices{2}; |
| std::vector<int> excluded_indices{-1, 5}; |
| TestMultiselectFieldDict(opt_array, values, selected_indices, |
| /*expected_use_indices=*/false, expected_indices, |
| excluded_indices); |
| } |
| } |