#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() здесь не очень помогает.