RichTextBox Сохранить "Направление выбора" - PullRequest
5 голосов
/ 27 сентября 2010

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

Я могу сохранить и загрузить свойства SelectionStart и SelectionLength, но не могу сохранить «направление выделения»:если пользователь выделял курсор вперед или назад от курсора.

Есть какие-либо идеи о том, как можно сохранить направление выделения или раскрасить текст без изменения выделения?

Ответы [ 3 ]

1 голос
/ 02 июля 2015

Я только что столкнулся с той же проблемой, и теперь я решил это с помощью EM_EXSETSEL.Когда cpMin> cpMax, он работает как «обратный выбор» (каретка в начале выделенного текста).Пока что я не нашел другого способа узнать текущее направление выбора (EM_EXGETSEL всегда возвращал cpMin

EDIT:

Это то, что яЯ использую, чтобы решить это.Возможно, есть какой-то более простой способ, но по крайней мере у меня работает следующее.

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);
}
1 голос
/ 27 сентября 2010

Тьфу, ужасная проблема.Нет, EM_SETPARAFORMAT может работать только с текущим выбором.И EM_EXSETSEL всегда ставит каретку в конце выделения.Вы можете определить направление выбора, наблюдая за изменением в SelectionStart, но вы не можете получить курсор в нужном месте.У элемента управления редактирования та же проблема.

Обычно это не проблема, поскольку перекрашивание происходит только тогда, когда пользователь изменяет текст, а не когда он выбирает текст.Единственный обходной путь, который я могу придумать, - это восстановление выделения путем нажатия клавиш.Это ужасно.

0 голосов
/ 09 мая 2012

Другим способом может быть установка свойства Rtf напрямую.Вам нужно знать синтаксис языка RTF.

PS.Это также сделает выбор недействительным.Я сделал инъекцию нажатия клавиш самостоятельно.

...