Использование QPDF с C # - PullRequest
0 голосов
/ 10 мая 2018

Я пытаюсь перевести эту команду qpdf:

qpdf --qdf --object-streams = отключить input.pdf editable.pdf

в эквивалентные вызовы методов, которые понадобятся мне при использовании библиотеки qpdf (доступно здесь: https://sourceforge.net/projects/qpdf/).

Я запустил dll qpdf через dumpbin, чтобы получить имена функций, и, просмотрев файлы заголовков, которые были включены для использования в проекте c ++, я могу увидеть параметры для функций.

Например, функция, необходимая для передачи описанной выше опции --object-streams, будет (насколько я могу судить) такой функцией:

void setObjectStreamMode(qpdf_object_stream_e);

из заголовочного файла c ++, который становится:

[DllImport("qpdf21.dll")]
static extern void _ZN10QPDFWriter19setObjectStreamModeE20qpdf_object_stream_e(int stateEnum);

в файле C #.

Проблема в том, что когда я использую вышеуказанную функцию, я получаю

AccessViolationException: попытка чтения или записи защищенной памяти

ошибка, из-за которой я думаю, что мне нужно каким-то образом создать объект QPDF, но я никогда не использовал объектно-ориентированные пинвоки, поэтому я не знаю, как сделать объект доступным в c #.

Если кто-то уже знаком с использованием dll в C # или даже в C ++ и может сказать мне правильные функции, которые необходимо вызвать для репликации команды, я был бы признателен за это!

Ответы [ 2 ]

0 голосов
/ 17 марта 2019

Похоже, вы нашли хороший ответ здесь. Вы обнаружили C API, который предназначен для помощи в использовании QPDF из языков, отличных от C ++, через DLL. C API в основном документируется в заголовочном файле qpdf-c.h . Вы также можете найти некоторую информацию в разделе Использование других языков руководства . C API не предоставляет полную функциональность библиотеки C ++ в qpdf. Если вы нашли недостающие фрагменты, пожалуйста, не стесняйтесь создавать проблему на github . Я пытаюсь обновить C API, когда добавляю новые интерфейсы, но я делаю это не для каждого интерфейса, и некоторые функции в CLI реализованы непосредственно в инструменте и не отображаются на одну библиотечную функцию.

Если C API недостаточно богат для вашего случая использования, также можно написать свой собственный класс C ++ с некоторыми функциями, объявленными extern "C", экспортировать их и создать дополнительную DLL, которую вы можете использовать так, как вам нужно. найдено выше. Вы можете посмотреть qpdf-c.cc как пример того, как это работает.

0 голосов
/ 10 мая 2018

Мне удалось выяснить это, приведенный ниже код повторяет команду, оказывается, я искал не тот заголовочный файл:

[DllImport("qpdf21.dll", CharSet = CharSet.Ansi, CallingConvention = CallingConvention.Cdecl)]
static extern IntPtr qpdf_init();

[DllImport("qpdf21.dll", CharSet = CharSet.Ansi, CallingConvention = CallingConvention.Cdecl)]
static extern void qpdf_cleanup(ref IntPtr qpdfData);

[DllImport("qpdf21.dll", CharSet = CharSet.Ansi, CallingConvention = CallingConvention.Cdecl)]
static extern int qpdf_read(IntPtr qpdfdata, [MarshalAs(UnmanagedType.LPStr)] string fileName, IntPtr password);

[DllImport("qpdf21.dll", CharSet = CharSet.Ansi, CallingConvention = CallingConvention.Cdecl)]
static extern void qpdf_set_object_stream_mode(IntPtr qpdf, int mode);

[DllImport("qpdf21.dll", CharSet = CharSet.Ansi, CallingConvention = CallingConvention.Cdecl)]
static extern void qpdf_set_qdf_mode(IntPtr qpdf, int value);

[DllImport("qpdf21.dll", CharSet = CharSet.Ansi, CallingConvention = CallingConvention.Cdecl)]
static extern int qpdf_init_write(IntPtr qpdf, [MarshalAs(UnmanagedType.LPStr)] string fileName);

[DllImport("qpdf21.dll", CharSet = CharSet.Ansi, CallingConvention = CallingConvention.Cdecl)]
static extern int qpdf_write(IntPtr qpdf);

static void Main(string[] args)
{
    //call init
    IntPtr qpdfData = qpdf_init();

    //call read (which gets and processes the input file)
    //0 result == good
    int result = qpdf_read(qpdfData, @"scan.pdf", IntPtr.Zero);

    //call init_write
    result = qpdf_init_write(qpdfData, @"scanEditable.pdf");

    //set write options
    //disable object stream mode
    qpdf_set_qdf_mode(qpdfData, 1);
    qpdf_set_object_stream_mode(qpdfData, 0);

    //call write
    result = qpdf_write(qpdfData);

    //call cleanup
    qpdf_cleanup(ref qpdfData);
}
...