Розфарбуйте різні частини рядка RichTextBox


109

Я намагаюся розфарбувати частини рядка, які потрібно додати до RichTextBox. У мене є струна, побудована з різних рядків.

string temp = "[" + DateTime.Now.ToShortTimeString() + "] " +
              userid + " " + message + Environment.NewLine;

Ось як виглядатиме повідомлення після його побудови.

[21:23] Користувач: моє повідомлення тут.

Я хочу, щоб все в дужках, включаючи дужки [9:23], було одним кольором, "користувач" - іншим кольором, а повідомлення - іншим кольором. Тоді я хотів би, щоб рядок додався до мого RichTextBox.

Як я можу це досягти?



5
Я здійснив пошук, і не знайшов нічого корисного.
Fatal510

Дякую за це просте рішення, добре працює для мене. Не забудьте використовувати додаток (...) щоразу, коли ви хочете додати текст, а також не використовувати оператор "+ =" або застосовані кольори.
Xhis

Відповіді:


238

Ось метод розширення, який перевантажує AppendTextметод кольоровим параметром:

public static class RichTextBoxExtensions
{
    public static void AppendText(this RichTextBox box, string text, Color color)
    {
        box.SelectionStart = box.TextLength;
        box.SelectionLength = 0;

        box.SelectionColor = color;
        box.AppendText(text);
        box.SelectionColor = box.ForeColor;
    }
}

І ось як би ви його використовували:

var userid = "USER0001";
var message = "Access denied";
var box = new RichTextBox
              {
                  Dock = DockStyle.Fill,
                  Font = new Font("Courier New", 10)
              };

box.AppendText("[" + DateTime.Now.ToShortTimeString() + "]", Color.Red);
box.AppendText(" ");
box.AppendText(userid, Color.Green);
box.AppendText(": ");
box.AppendText(message, Color.Blue);
box.AppendText(Environment.NewLine);

new Form {Controls = {box}}.ShowDialog();

Зауважте, що ви можете помітити деяке мерехтіння, якщо ви видаєте багато повідомлень. Дивіться цю статтю C # Corner, щоб дізнатися, як зменшити мерехтіння RichTextBox.


3
У мене були проблеми з цим. Я використовував в іншому місці box.Text += mystringі тому всі кольори зникли. Коли я замінив це box.AppendText(mystring), це спрацювало як принадність.
Натрій

3
У мене виникають проблеми з кодом, колір зникає при додаванні рядка в якийсь інший колір. Єдина відмінність полягає в тому, що я призначаю var box до раніше зробленого richtextbox ....
manu_dilip_shah

Що я роблю неправильно ... "SelectionStart" не є властивістю RichTextBox (або я не можу отримати доступ до нього принаймні)? Я знайшов "Вибір", але це власність лише для отримання ...
00jt

2
Це спеціально для WinForms. Ви використовуєте WPF випадково?
Натан Баулч,

У мене немає перевантаження ikee цього, тільки AppendText(string text)з WinForms
vaso123

12

Я розширив метод як шрифт як параметр:

public static void AppendText(this RichTextBox box, string text, Color color, Font font)
{
    box.SelectionStart = box.TextLength;
    box.SelectionLength = 0;

    box.SelectionColor = color;
    box.SelectionFont = font;
    box.AppendText(text);
    box.SelectionColor = box.ForeColor;
}

Примітка - для цього потрібно використовувати метод AppendText. Якщо присвоїти що-небудь властивості тексту поля, це порушить його.
Джефф

9

Це модифікована версія, яку я вклав у свій код (я використовую .Net 4.5), але, думаю, він також повинен працювати на 4.0.

public void AppendText(string text, Color color, bool addNewLine = false)
{
        box.SuspendLayout();
        box.SelectionColor = color;
        box.AppendText(addNewLine
            ? $"{text}{Environment.NewLine}"
            : text);
        box.ScrollToCaret();
        box.ResumeLayout();
}

Відмінності від оригіналу:

  • можливість додати текст до нового рядка або просто додати його
  • не потрібно змінювати вибір, він працює так само
  • вставлено ScrollToCaret для примусового автоматичного прокручування
  • додано призупинення / відновлення розмовних розмов

6

Я думаю, що зміна "вибраного тексту" в RichTextBox не є правильним способом додавання кольорового тексту. Отже, тут спосіб додати "кольоровий блок":

        Run run = new Run("This is my text");
        run.Foreground = new SolidColorBrush(Colors.Red); // My Color
        Paragraph paragraph = new Paragraph(run);
        MyRichTextBlock.Document.Blocks.Add(paragraph);

Від MSDN :

Властивість Blocks є властивістю вмісту RichTextBox. Це колекція елементів абзацу. Вміст кожного елемента абзацу може містити такі елементи:

  • В лінію

  • InlineUIContainer

  • Біжи

  • Проміжок

  • Сміливий

  • Гіперпосилання

  • Курсив

  • Підкресли

  • LineBreak

Тому я думаю, що вам потрібно розділити рядок залежно від кольору частин та створити стільки Runоб'єктів, скільки потрібно.


Я сподівався, що це відповідь, яку я справді шукав, але, здається, це відповідь WPF, а не відповідь WinForms.
Крістен Хаммак

3

Це працює для мене! Я сподіваюся, що вам це стане в нагоді!

public static RichTextBox RichTextBoxChangeWordColor(ref RichTextBox rtb, string startWord, string endWord, Color color)
{
    rtb.SuspendLayout();
    Point scroll = rtb.AutoScrollOffset;
    int slct = rtb.SelectionIndent;
    int ss = rtb.SelectionStart;
    List<Point> ls = GetAllWordsIndecesBetween(rtb.Text, startWord, endWord, true);
    foreach (var item in ls)
    {
        rtb.SelectionStart = item.X;
        rtb.SelectionLength = item.Y - item.X;
        rtb.SelectionColor = color;
    }
    rtb.SelectionStart = ss;
    rtb.SelectionIndent = slct;
    rtb.AutoScrollOffset = scroll;
    rtb.ResumeLayout(true);
    return rtb;
}

public static List<Point> GetAllWordsIndecesBetween(string intoText, string fromThis, string toThis,bool withSigns = true)
{
    List<Point> result = new List<Point>();
    Stack<int> stack = new Stack<int>();
    bool start = false;
    for (int i = 0; i < intoText.Length; i++)
    {
        string ssubstr = intoText.Substring(i);
        if (ssubstr.StartsWith(fromThis) && ((fromThis == toThis && !start) || !ssubstr.StartsWith(toThis)))
        {
            if (!withSigns) i += fromThis.Length;
            start = true;
            stack.Push(i);
        }
        else if (ssubstr.StartsWith(toThis) )
        {
            if (withSigns) i += toThis.Length;
            start = false;
            if (stack.Count > 0)
            {
                int startindex = stack.Pop();
                result.Add(new Point(startindex,i));
            }
        }
    }
    return result;
}

0

Вибір тексту, як хтось сказав, може виділитись миттєво. В Windows Forms applicationsне існує ніяких інших рішень для цієї проблеми, але сьогодні я знайшов погано, працює, спосіб вирішити: ви можете покласти PictureBoxвнахлест на RichtextBoxз скріншотом , якщо, під час вибору і зміни кольору або шрифту, що робить його після того, як знову з’являються всі, коли операція завершена.

Код тут ...

//The PictureBox has to be invisible before this, at creation
//tb variable is your RichTextBox
//inputPreview variable is your PictureBox
using (Graphics g = inputPreview.CreateGraphics())
{
    Point loc = tb.PointToScreen(new Point(0, 0));
    g.CopyFromScreen(loc, loc, tb.Size);
    Point pt = tb.GetPositionFromCharIndex(tb.TextLength);
    g.FillRectangle(new SolidBrush(Color.Red), new Rectangle(pt.X, 0, 100, tb.Height));
}
inputPreview.Invalidate();
inputPreview.Show();
//Your code here (example: tb.Select(...); tb.SelectionColor = ...;)
inputPreview.Hide();

Краще використовувати WPF; це рішення не є ідеальним, але для Winform воно працює.


0
private void Log(string s , Color? c = null)
        {
            richTextBox.SelectionStart = richTextBox.TextLength;
            richTextBox.SelectionLength = 0;
            richTextBox.SelectionColor = c ?? Color.Black;
            richTextBox.AppendText((richTextBox.Lines.Count() == 0 ? "" : Environment.NewLine) + DateTime.Now + "\t" + s);
            richTextBox.SelectionColor = Color.Black;

        }

0

Використовуючи Selection у WPF, агрегуючи з декількох інших відповідей, не потрібен інший код (крім перерахунку важкості та функції GetSeverityColor)

 public void Log(string msg, Severity severity = Severity.Info)
    {
        string ts = "[" + DateTime.Now.ToString("HH:mm:ss") + "] ";
        string msg2 = ts + msg + "\n";
        richTextBox.AppendText(msg2);

        if (severity > Severity.Info)
        {
            int nlcount = msg2.ToCharArray().Count(a => a == '\n');
            int len = msg2.Length + 3 * (nlcount)+2; //newlines are longer, this formula works fine
            TextPointer myTextPointer1 = richTextBox.Document.ContentEnd.GetPositionAtOffset(-len);
            TextPointer myTextPointer2 = richTextBox.Document.ContentEnd.GetPositionAtOffset(-1);

            richTextBox.Selection.Select(myTextPointer1,myTextPointer2);
            SolidColorBrush scb = new SolidColorBrush(GetSeverityColor(severity));
            richTextBox.Selection.ApplyPropertyValue(TextElement.BackgroundProperty, scb);

        }

        richTextBox.ScrollToEnd();
    }

0

Я створив цю функцію після дослідження в Інтернеті, оскільки я хотів надрукувати рядок XML, коли ви вибираєте рядок із подання сітки даних.

static void HighlightPhrase(RichTextBox box, string StartTag, string EndTag, string ControlTag, Color color1, Color color2)
{
    int pos = box.SelectionStart;
    string s = box.Text;
    for (int ix = 0; ; )
    {
        int jx = s.IndexOf(StartTag, ix, StringComparison.CurrentCultureIgnoreCase);
        if (jx < 0) break;
        int ex = s.IndexOf(EndTag, ix, StringComparison.CurrentCultureIgnoreCase);
        box.SelectionStart = jx;
        box.SelectionLength = ex - jx + 1;
        box.SelectionColor = color1;
        
        int bx = s.IndexOf(ControlTag, ix, StringComparison.CurrentCultureIgnoreCase);
        int bxtest = s.IndexOf(StartTag, (ex + 1), StringComparison.CurrentCultureIgnoreCase);
        if (bx == bxtest)
        {
            box.SelectionStart = ex + 1;
            box.SelectionLength = bx - ex + 1;
            box.SelectionColor = color2;
        }
        
        ix = ex + 1;
    }
    box.SelectionStart = pos;
    box.SelectionLength = 0;
}

і ось так ви це називаєте

   HighlightPhrase(richTextBox1, "<", ">","</", Color.Red, Color.Black);

Це не вирішує дану проблему, лише ілюструє, як рішення може працювати.
Йозеф Блаха
Використовуючи наш веб-сайт, ви визнаєте, що прочитали та зрозуміли наші Політику щодо файлів cookie та Політику конфіденційності.
Licensed under cc by-sa 3.0 with attribution required.