Как вы не позволяете RichTextBox обновлять свой дисплей?

#c# #winforms #richtextbox

Вопрос:

У меня есть RichTextBox, в котором мне нужно часто обновлять свойство Text, но когда я это делаю, RichTextBox раздражающе «мигает», поскольку он обновляется на протяжении всего вызова метода.

Я надеялся найти простой способ временно отключить обновление экрана до тех пор, пока мой метод не будет завершен, но единственное, что я нашел в Интернете, — это переопределить метод WndProc. Я использовал этот подход, но с некоторыми трудностями и побочными эффектами, и это также усложняет отладку. Просто кажется, что должен быть лучший способ сделать это. Может ли кто-нибудь указать мне на лучшее решение?

Ответ №1:

Вот полный и рабочий пример:

     private const int WM_USER = 0x0400;
    private const int EM_SETEVENTMASK = (WM_USER   69);
    private const int WM_SETREDRAW = 0x0b;
    private IntPtr OldEventMask;       

    [DllImport("user32.dll", CharSet=CharSet.Auto)]
    private static extern IntPtr SendMessage(IntPtr hWnd, int msg, IntPtr wParam, IntPtr lParam);

    public void BeginUpdate()
    {
        SendMessage(this.Handle, WM_SETREDRAW, IntPtr.Zero, IntPtr.Zero);
        OldEventMask = (IntPtr)SendMessage(this.Handle, EM_SETEVENTMASK, IntPtr.Zero, IntPtr.Zero);
    }       

    public void EndUpdate()
    {
        SendMessage(this.Handle, WM_SETREDRAW, (IntPtr)1, IntPtr.Zero);
        SendMessage(this.Handle, EM_SETEVENTMASK, IntPtr.Zero, OldEventMask);
    }
 

Ответ №2:

Я задал первоначальный вопрос, и ответ, который лучше всего сработал для меня, заключался в использовании BoltBait функции SendMessage() с WM_SETREDRAW. Похоже, у него меньше побочных эффектов, чем при использовании метода WndProc, и в моем приложении он работает в два раза быстрее, чем LockWindowUpdate.

В моем расширенном классе RichTextBox я только что добавил эти два метода и вызываю их всякий раз, когда мне нужно остановить перезапуск перерисовки, пока я выполняю некоторую обработку. Если бы я хотел сделать это за пределами класса RichTextBox, я думаю, что это сработало бы, просто заменив «это» ссылкой на ваш экземпляр RichTextBox.

     private void StopRepaint()
    {
        // Stop redrawing:
        SendMessage(this.Handle, WM_SETREDRAW, 0, IntPtr.Zero);
        // Stop sending of events:
        eventMask = SendMessage(this.Handle, EM_GETEVENTMASK, 0, IntPtr.Zero);
    }

    private void StartRepaint()
    {
        // turn on events
        SendMessage(this.Handle, EM_SETEVENTMASK, 0, eventMask);
        // turn on redrawing
        SendMessage(this.Handle, WM_SETREDRAW, 1, IntPtr.Zero);
        // this forces a repaint, which for some reason is necessary in some cases.
        this.Invalidate();
    }
 

Комментарии:

1. Есть ли шанс завершить код? Это почти не поддается компиляции, как написано в настоящее время.

2. Ожидается вызов функции Invalidate (). Я использовал вариант этого в приложении MDI, где каждое из дочерних окон MDI содержало поле RichTextBox. В этом случае мне пришлось аннулировать контроль с помощью child.richTextBox.Invalidate();

Ответ №3:

Найдено здесь: http://bytes.com/forum/thread276845.html

В итоге я отправил WM_SETREDRAW через SendMessage, чтобы отключить его, а затем повторно включить, после чего я завершил обновление с помощью функции Invalidate (). Похоже, это сработало.

Я никогда не пробовал этот метод. Я написал приложение с RTB, которое имеет подсветку синтаксиса, и использовал следующее в классе RTB:

 protected override void WndProc(ref Message m)
{
    if (m.Msg == paint)
    {
        if (!highlighting)
        {
            base.WndProc(ref m); // if we decided to paint this control, just call the RichTextBox WndProc
        }
        else
        {
            m.Result = IntPtr.Zero; // not painting, must set this to IntPtr.Zero if not painting otherwise serious problems.
        }
    }
    else
    {
        base.WndProc(ref m); // message other than paint, just do what you normally do.
    }
}
 

Надеюсь, это поможет.

Комментарии:

1. WM_SETREDRAW работал на меня. Должно быть то же самое, что и LockWindowUpdate.

Ответ №4:

Не могли бы вы просто сохранить текст в строку, выполнить свои манипуляции со строкой и в конце метода сохранить его обратно в свойство Text?

Ответ №5:

Я бы посоветовал посмотреть LockWindowUpdate

 
[DllImport("user32.dll", EntryPoint="LockWindowUpdate", SetLastError=true,
ExactSpelling=true, CharSet=CharSet.Auto,
CallingConvention=CallingConvention.StdCall)]
 

Комментарии:

1. -1. Это неправильное его использование. Видишь blogs.msdn.com/b/oldnewthing/archive/2007/02/19/1716211.aspx .

Ответ №6:

Попробуйте это:

 myRichTextBox.SuspendLayout();
DoStuff();
myRichTextBox.ResumeLayout();
 

Комментарии:

1. возможно, вам также придется добавить myRichTextBox.Включено = ложь; и позднее Включено = истина;

2. Функция SuspendLayout() здесь не очень помогает.