Определить текущую страницу в объекте AxAcroPDF в C#

Элемент управления Adobe AxAcroPDF не имеет функции для возврата номера текущей страницы. Я просто создаю личную утилиту, поэтому мне посчастливилось воспользоваться хакерским методом, который, как я думал, я бы поделился... Он исправлен вместе с кусочками по всей сети. Он использует собственные функции Windows из User32.dll для перечисления дочерних элементов элемента управления и поиска текстового поля на панели инструментов, соответствующего номеру страницы. Текст из этого затем читается с использованием вызова SendMessage.

1 ответ

Этот метод перечисляет дочерние компоненты Adobe Viewer PDF и находит текстовые поля на панели инструментов. Одно из текстовых полей - номер страницы, одно - текущее значение масштабирования. Текстовое поле без "%" в его содержимом принимается как текстовое поле номера страницы. Содержимое текстового поля извлекается с помощью функции SendMessage.

Возможно, вам сначала потребуется вызвать SetToolbarVisible (true) для компонента средства просмотра, чтобы убедиться, что панель инструментов (и, следовательно, текстовые поля) видимы.

Это ужасное и хакерское решение, которое может легко сломаться, когда Adobe обновит программу просмотра. Было бы здорово, если бы Adobe добавила метод "getCurrentPage", чтобы всего этого можно было избежать.

  //you can get the handle parameter for this method as: yourPDFControl.Handle
  public static string GetPageNumber(IntPtr adobeViewerHandle)
    {
        //get a list of all windows held by parent 
        List<IntPtr> childrenWindows = new List<IntPtr>();
        GCHandle listHandle = GCHandle.Alloc(childrenWindows);
        try
        {
            EnumWindowProc childProc = new EnumWindowProc(EnumWindow);
            EnumChildWindows(adobeViewerHandle, childProc, GCHandle.ToIntPtr(listHandle));
        }
        finally
        {
            if (listHandle.IsAllocated)
                listHandle.Free();
        }

        //now have a list of the children, look for text boxes with class name "Edit"
        for (int i = 0; i < childrenWindows.Count; i++)
        {
            int nRet;
            // Pre-allocate 256 characters, the maximum class name length.
            StringBuilder ClassName = new StringBuilder(256);
            //Get the window class name
            nRet = GetClassName(childrenWindows.ElementAt(i), ClassName, ClassName.Capacity);

            if (ClassName.ToString().CompareTo("Edit") == 0)
            {
                IntPtr resultPointer = Marshal.AllocHGlobal(200);
                StringBuilder text = new StringBuilder(20);
                NativeMethods.SendMessage(childrenWindows.ElementAt(i), 0x000D, text.Capacity, text); //0x000D is WM_GETTEXT message
                if (text.ToString().Contains("%")) //we don't want the text box for the PDF scale (e.g. 66.7% zoomed etc.)
                {
                    continue;
                } else
                {
                    return text.ToString(); // the only other text box is the page number box
                }
            }
        }

        //Note I return as a string because PDF supports page labels, "I", "ii", "iv" etc. or even section labels "A", "B". So you're not guaranteed a numerical page number.
        return "0";

    }

    private static bool EnumWindow(IntPtr handle, IntPtr pointer)
    {
        GCHandle gch = GCHandle.FromIntPtr(pointer);
        List<IntPtr> list = gch.Target as List<IntPtr>;
        if (list == null)
            throw new InvalidCastException("GCHandle Target could not be cast as List<IntPtr>");

        list.Add(handle);
        return true;
    }


        [DllImport("user32.dll")]
        [return: MarshalAs(UnmanagedType.Bool)]
        internal static extern bool EnumChildWindows(IntPtr window,
                                                        EnumWindowProc callback,
                                                        IntPtr i);

        internal delegate bool EnumWindowProc(IntPtr hWnd, IntPtr parameter);

        [DllImport("user32.dll", SetLastError = true, CharSet = CharSet.Auto)]
        internal static extern int GetWindowTextLength(IntPtr hWnd);

        [DllImport("user32.dll", SetLastError = true, CharSet = CharSet.Auto)]
        internal static extern int GetWindowText(IntPtr hWnd,
                                                   StringBuilder lpString,
                                                   int nMaxCount);

        [DllImport("user32.dll", SetLastError = true, CharSet = CharSet.Auto)]
        internal static extern int GetClassName(IntPtr hWnd, StringBuilder lpClassName, int nMaxCount);

        [DllImport("user32.dll", SetLastError = true, CharSet = CharSet.Auto)]
        internal static extern bool SendMessage(IntPtr hWnd, uint Msg, int wParam, StringBuilder lParam);
Другие вопросы по тегам