0

我是 PDFNet SDK 的新手,仍在尝试拼凑一些 PDF 功能的工作原理。SDK 的广度和深度给我留下了深刻的印象,但我试图了解 SDK 如何与层(也称为可选内容组或 OCG)一起工作。特别是,我试图确定如何检测哪些图层是不可见的(关闭),以便可以从 PDF 中删除它们及其内容。我最初专注于 pdftron.PDF.OCG 命名空间中的类,尤其是 Group 类。这允许我迭代组并获取基本信息,例如名称和当前状态 (ON/OFF),如下面的代码片段所示:

Config init_cfg = doc.GetOCGConfig(); 
Context ctx = new Context(init_cfg); 


Obj ocgs = doc.GetOCGs(); // Get the array of all OCGs in the document. 
if (ocgs != null) 
{ 
    int i, sz = ocgs.Size(); 
    for (i = 0; i < sz; ++i) 
    { 
        Group ocg = new Group(ocgs.GetAt(i)); 
        bool ocgState = ocg.GetCurrentState(ctx);  // check if ocg is OFF or ON 
        if (!ocgState) // layer is not visible
        {
            // need calls here to delete layer and content
        }
    } 
} 

到目前为止,一切都很好。但是,我看不到有任何方法可以删除这些类中的图层和相关内容。我错过了什么吗?如果没有,SDK中是否有另一种方法来完成删除层?

提前致谢!

4

1 回答 1

0

好的,在对 SDK 示例和大量实验进行了更多研究之后,我找到了一种迄今为止适用于我尝试过的 PDF 的方法。该解决方案基于 ElementEdit 示例,并结合了 Context.SetNonOCDrawing()、Context.SetOCDrawMode() 和 Element.IsOCVisible() 方法,以根据图层的 ON/OFF 状态省略不可见的页面元素. 最终结果是删除了关闭的图层的内容,图层本身仍然留在后面,但现在是空的。不完美,但足以满足我的目的。无论如何,这是我根据 SDK 示例提出的代码:

    private void ProcessElements(ElementReader reader, ElementWriter writer, XSet visited)
    {
        Element element;
        while ((element = reader.Next()) != null)   // Read page contents
        {

            switch (element.GetType())
            {
                //always write these element types (visible or non-visible)
                case Element.Type.e_marked_content_begin:
                case Element.Type.e_marked_content_end:
                case Element.Type.e_marked_content_point:
                case Element.Type.e_group_begin:
                case Element.Type.e_group_end:
                    writer.WriteElement(element);
                    continue;
            }

            // do not retain other elements that are not visible
            if (!element.IsOCVisible())
                continue;

            // retain remaining element types if visible
            switch (element.GetType())
            {

                // special processing for form elements
                case Element.Type.e_form:
                    {
                        writer.WriteElement(element); // write Form XObject reference to current stream

                        Obj form_obj = element.GetXObject();
                        if (!visited.Contains(form_obj.GetObjNum())) // if this XObject has not been processed
                        {
                            // recursively process the Form XObject
                            visited.Add(form_obj.GetObjNum());
                            ElementWriter new_writer = new ElementWriter();

                            reader.FormBegin();
                            new_writer.Begin(form_obj, true);
                            ProcessElements(reader, new_writer, visited);
                            new_writer.End();
                            reader.End();
                        }
                        break;
                    }

                default:
                    writer.WriteElement(element);
                    break;
            }
        }
    }


    private void RemoveNonVisibleLayers_Click(string filename, string outputfile)
    {

        PDFNet.Initialize();

        try
        {

            // Open the test file
            XSet visited = new XSet();

            using (PDFDoc doc = new PDFDoc(filename))
            using (ElementReader page_reader = new ElementReader())
            using (ElementWriter writer = new ElementWriter())
            {
                doc.InitSecurityHandler();
                Config init_cfg = doc.GetOCGConfig();
                Context ctx = new Context(init_cfg);

                ctx.SetNonOCDrawing(false);
                ctx.SetOCDrawMode(Context.OCDrawMode.e_VisibleOC);

                PageIterator itr;
                for (itr = doc.GetPageIterator(); itr.HasNext(); itr.Next())        //  Read every page
                {
                    itr.GetPageNumber());

                    page_reader.Begin(itr.Current(), ctx);
                    writer.Begin(itr.Current(), ElementWriter.WriteMode.e_replacement, false);

                    ProcessElements(page_reader, writer, visited);
                    writer.End();
                    page_reader.End();
                }

                doc.Save(outputfile, SDFDoc.SaveOptions.e_remove_unused);
                doc.Close();
            }
        }
        catch (PDFNetException ee)
        {
            Console.WriteLine(ee.Message);
        }

    }
于 2015-08-18T16:51:58.960 回答