我有一个 WinForms 程序,每当您更改选择时,RichTextBox都需要更改某些其他文本的颜色。为了做到这一点,它必须选择该文本,因此我失去了当前的选择。
我可以保存并加载SelectionStart和SelectionLength属性,但是我不能保持“选择方向”:如果用户从光标向前或向后突出显示。
有关如何保存选择方向或为文本着色而不必更改选择的任何想法?
答案 0 :(得分:1)
哎呀,丑陋的问题。不,EM_SETPARAFORMAT只能用于当前选择。并且EM_EXSETSEL始终将插入符号放在选择的末尾。您可以通过观察SelectionStart中的更改来检测选择方向,但是您无法将插入符号放在正确的位置。编辑控件也有同样的问题。
这通常不是问题,因为重新着色仅在用户修改文本时发生,而不是在她选择文本时。我能想到的唯一解决方法是通过注入击键来恢复选择。那太难了。
答案 1 :(得分:1)
我遇到了同样的问题,现在我通过使用EM_EXSETSEL解决了这个问题。当cpMin> cpMax,它就像"向后选择" (所选文本开头的插入符号)。然而,我还没有找到任何其他方法来找出当前的选择方向(EM_EXGETSEL总是返回cpMin< cpMax)但是在SelectionStart / Length更改后......
编辑:
这就是我用来解决这个问题的方法。可能有一些更简单的方法,但至少以下方法适用于我。
using System.Runtime.InteropServices;
//********************
//SendMessage stuff for EM_EXSETSEL
//********************
[StructLayout(LayoutKind.Sequential)]
public struct CHARRANGE
{
public int cpMin;
public int cpMax;
}
[DllImport("user32.dll")]
private static extern IntPtr SendMessage(IntPtr hWnd, UInt32 msg, IntPtr wParam, ref CHARRANGE lParam);
private const UInt32 WM_USER = 0x0400;
private const UInt32 EM_EXSETSEL = WM_USER + 55;
private const UInt32 EM_EXGETSEL = WM_USER + 52;
//********************
//event handlers
//********************
//locking variable to avoid stack overflow while setting selection in code
private bool richTextBox1_SelectionChanged_lock = false;
//handler for richTextBox selection change event
private void richTextBox1_SelectionChanged(object sender, EventArgs e)
{
if (richTextBox1_SelectionChanged_lock) return;
richTextBox1_SelectionChanged_lock = true;
//detect selection changes and store information needed for restoring
TrackRTBSelection(richTextBox1.SelectionStart, richTextBox1.SelectionLength);
//here do whatever you want with selection (some analysis to show font name in font selection comboBox etc.)
//...
//restore selection from saved informations
SetRTBSelectionBasedOnTracking();
richTextBox1_SelectionChanged_lock = false;
}
//sample button click handler for changing fore color of selected text
private void buttonSetForeColor_Click(object sender, EventArgs e)
{
if (colorDialog1.ShowDialog() == DialogResult.Cancel)
return;
//prevent selection change events while we are changing font colors
if (richTextBox1_SelectionChanged_lock) return;
richTextBox1_SelectionChanged_lock = true;
//save selection parameters for use in loop
int selStart = richTextBox1.SelectionStart;
int selLength = richTextBox1.SelectionLength;
for (int i = 0; i < selLength; i++)
{
richTextBox1.SelectionLength = 1;
richTextBox1.SelectionStart = selStart + i;
richTextBox1.SelectionColor = colorDialog1.Color;
}
//restore selection from saved informations
SetRTBSelectionBasedOnTracking();
richTextBox1_SelectionChanged_lock = false;
}
//********************
//selection tracking utilities
//********************
//false - caret at the beginning; true - caret at the end
private bool caretPosition = false;
private int lastSelectionStart = -1;
private int lastSelectionLength = -1;
//initialize selection informations. this must be called during Form_Load
private void InitRTBSelection()
{
richTextBox1.SelectionStart = 0;
richTextBox1.SelectionLength = 0;
caretPosition = false;
lastSelectionStart = 0;
lastSelectionLength = 0;
//force "selection changed" to detect "selection changes" for the first time
richTextBox1_SelectionChanged(richTextBox1, new EventArgs());
}
//this method detects changes in selection, based on selection parameters received from richTextBox
private void TrackRTBSelection(int newSelectionStart, int newSelectionLength)
{
int condition = 0;
int s_change = (newSelectionStart - lastSelectionStart > 0) ?
1 :
(newSelectionStart - lastSelectionStart < 0) ? -1 : 0;
int l_change = (newSelectionLength - lastSelectionLength > 0) ?
1 :
(newSelectionLength - lastSelectionLength < 0) ? -1 : 0;
//these conditions where created over change table for all user-achievable scenarios
condition = (newSelectionLength == 0 ||
(l_change == 1 && s_change == -1) ||
(l_change == -1 && s_change == 1 && caretPosition == false)) ? 1 : condition;
condition = (s_change == 0 && (l_change == 1 || (caretPosition == true && l_change == -1))) ? 2 : condition;
switch (condition)
{
case 1: caretPosition = false; break;
case 2: caretPosition = true; break;
default: break; //if no condition was satisfied then maintain current information
}
lastSelectionStart = newSelectionStart;
lastSelectionLength = newSelectionLength;
}
//set richTextBox selection using EM_EXSETSEL
private void SetRTBSelectionBasedOnTracking()
{
CHARRANGE chrrange = new CHARRANGE
{
cpMin = caretPosition ? lastSelectionStart : lastSelectionStart + lastSelectionLength,
cpMax = caretPosition ? lastSelectionStart + lastSelectionLength : lastSelectionStart
};
SendMessage(richTextBox1.Handle, EM_EXSETSEL, IntPtr.Zero, ref chrrange);
}
答案 2 :(得分:0)
另一种方法是直接设置Rtf属性。您需要知道rtf语言的语法。
PS。这也将使选择无效。我自己做了击键注射。