Sign Up

Sign Up to our social questions and Answers Engine to ask questions, answer people’s questions, and connect with other people.

Have an account? Sign In

Have an account? Sign In Now

Sign In

Login to our social questions & Answers Engine to ask questions answer people’s questions & connect with other people.

Sign Up Here

Forgot Password?

Don't have account, Sign Up Here

Forgot Password

Lost your password? Please enter your email address. You will receive a link and will create a new password via email.

Have an account? Sign In Now

You must login to ask a question.

Forgot Password?

Need An Account, Sign Up Here

Please briefly explain why you feel this question should be reported.

Please briefly explain why you feel this answer should be reported.

Please briefly explain why you feel this user should be reported.

Sign InSign Up

The Archive Base

The Archive Base Logo The Archive Base Logo

The Archive Base Navigation

  • SEARCH
  • Home
  • About Us
  • Blog
  • Contact Us
Search
Ask A Question

Mobile menu

Close
Ask a Question
  • Home
  • Add group
  • Groups page
  • Feed
  • User Profile
  • Communities
  • Questions
    • New Questions
    • Trending Questions
    • Must read Questions
    • Hot Questions
  • Polls
  • Tags
  • Badges
  • Buy Points
  • Users
  • Help
  • Buy Theme
  • SEARCH
Home/ Questions/Q 6605435
In Process

The Archive Base Latest Questions

Editorial Team
  • 0
Editorial Team
Asked: May 25, 20262026-05-25T19:17:01+00:00 2026-05-25T19:17:01+00:00

I have a TextBlock in my WPF window. <TextBlock> Some <Bold>formatted</Bold> text. </TextBlock> When

  • 0

I have a TextBlock in my WPF window.

 <TextBlock>
     Some <Bold>formatted</Bold> text.
 </TextBlock>

When it is rendered it looks like this,

Some formatted text.

My question is, can I bind this inline “content” to a resource in my application?

I got as far as:

Making an application resource string,

myText="Some <Bold>formatted</Bold> text."

and the following xaml (Some code omitted for brevity)

 <Window xmlns:props="clr-namespace:MyApp.Properties">
     <Window.Resources>
         <props:Resources x:Key="Resources"/>
     </Window.Resources>
      <TextBlock x:Name="Try1" 
          Text="{Binding Source={StaticResource Resources} Path=myText}"/>
     <TextBlock x:Name="Try2">
          <Binding Source="{StaticResource Resources}" Path="myText" />
     </TextBlock>
 </Window>

Try1 renders with the tags in place and not effecting formatting.

Some <Bold>formatted<Bold> text.

Try2 will not compile or render because the resource “myText” is not of type Inline but a string.

Is this seemingly simple task possible and if so how?

  • 1 1 Answer
  • 0 Views
  • 0 Followers
  • 0
Share
  • Facebook
  • Report

Leave an answer
Cancel reply

You must login to add an answer.

Forgot Password?

Need An Account, Sign Up Here

1 Answer

  • Voted
  • Oldest
  • Recent
  • Random
  1. Editorial Team
    Editorial Team
    2026-05-25T19:17:02+00:00Added an answer on May 25, 2026 at 7:17 pm

    Here is my modified code for recursively format text. It handles Bold, Italic, Underline and LineBreak but can easily be extended to support more (modify the switch statement).

    public static class MyBehavior
    {
        public static string GetFormattedText(DependencyObject obj)
        {
            return (string)obj.GetValue(FormattedTextProperty);
        }
    
        public static void SetFormattedText(DependencyObject obj, string value)
        {
            obj.SetValue(FormattedTextProperty, value);
        }
    
        public static readonly DependencyProperty FormattedTextProperty =
            DependencyProperty.RegisterAttached("FormattedText",
            typeof(string),
            typeof(MyBehavior),
            new UIPropertyMetadata("", FormattedTextChanged));
    
        static Inline Traverse(string value)
        {
            // Get the sections/inlines
            string[] sections = SplitIntoSections(value);
    
            // Check for grouping
            if (sections.Length.Equals(1))
            {
                string section = sections[0];
                string token; // E.g <Bold>
                int tokenStart, tokenEnd; // Where the token/section starts and ends.
    
                // Check for token
                if (GetTokenInfo(section, out token, out tokenStart, out tokenEnd))
                {
                    // Get the content to further examination
                    string content = token.Length.Equals(tokenEnd - tokenStart) ?
                        null :
                        section.Substring(token.Length, section.Length - 1 - token.Length * 2);
    
                    switch (token)
                    {
                        case "<Bold>":
                            return new Bold(Traverse(content));
                        case "<Italic>":
                            return new Italic(Traverse(content));
                        case "<Underline>":
                            return new Underline(Traverse(content));
                        case "<LineBreak/>":
                            return new LineBreak();
                        default:
                            return new Run(section);
                    }
                }
                else return new Run(section);
            }
            else // Group together
            {
                Span span = new Span();
    
                foreach (string section in sections)
                    span.Inlines.Add(Traverse(section));
    
                return span;
            }
        }
    
        /// <summary>
        /// Examines the passed string and find the first token, where it begins and where it ends.
        /// </summary>
        /// <param name="value">The string to examine.</param>
        /// <param name="token">The found token.</param>
        /// <param name="startIndex">Where the token begins.</param>
        /// <param name="endIndex">Where the end-token ends.</param>
        /// <returns>True if a token was found.</returns>
        static bool GetTokenInfo(string value, out string token, out int startIndex, out int endIndex)
        {
            token = null;
            endIndex = -1;
    
            startIndex = value.IndexOf("<");
            int startTokenEndIndex = value.IndexOf(">");
    
            // No token here
            if (startIndex < 0)
                return false;
    
            // No token here
            if (startTokenEndIndex < 0)
                return false;
    
            token = value.Substring(startIndex, startTokenEndIndex - startIndex + 1);
    
            // Check for closed token. E.g. <LineBreak/>
            if (token.EndsWith("/>"))
            {
                endIndex = startIndex + token.Length;
                return true;
            }
    
            string endToken = token.Insert(1, "/");
    
            // Detect nesting;
            int nesting = 0;
            int temp_startTokenIndex = -1;
            int temp_endTokenIndex = -1;
            int pos = 0;
            do
            {
                temp_startTokenIndex = value.IndexOf(token, pos);
                temp_endTokenIndex = value.IndexOf(endToken, pos);
    
                if (temp_startTokenIndex >= 0 && temp_startTokenIndex < temp_endTokenIndex)
                {
                    nesting++;
                    pos = temp_startTokenIndex + token.Length;
                }
                else if (temp_endTokenIndex >= 0 && nesting > 0)
                {
                    nesting--;
                    pos = temp_endTokenIndex + endToken.Length;
                }
                else // Invalid tokenized string
                    return false;
    
            } while (nesting > 0);
    
            endIndex = pos;
    
            return true;
        }
    
        /// <summary>
        /// Splits the string into sections of tokens and regular text.
        /// </summary>
        /// <param name="value">The string to split.</param>
        /// <returns>An array with the sections.</returns>
        static string[] SplitIntoSections(string value)
        {
            List<string> sections = new List<string>();
    
            while (!string.IsNullOrEmpty(value))
            {
                string token;
                int tokenStartIndex, tokenEndIndex;
    
                // Check if this is a token section
                if (GetTokenInfo(value, out token, out tokenStartIndex, out tokenEndIndex))
                {
                    // Add pretext if the token isn't from the start
                    if (tokenStartIndex > 0)
                        sections.Add(value.Substring(0, tokenStartIndex));
    
                    sections.Add(value.Substring(tokenStartIndex, tokenEndIndex - tokenStartIndex));
                    value = value.Substring(tokenEndIndex); // Trim away
                }
                else
                { // No tokens, just add the text
                    sections.Add(value);
                    value = null;
                }
            }
    
            return sections.ToArray();
        }
    
        private static void FormattedTextChanged(DependencyObject sender, DependencyPropertyChangedEventArgs e)
        {
            string value = e.NewValue as string;
    
            TextBlock textBlock = sender as TextBlock;
    
            if (textBlock != null)
                textBlock.Inlines.Add(Traverse(value));
        }
    }
    

    Edit: (proposed by Spook)

    A shorter version, but requires the text to be XML-valid:

    using System.Windows;
    using System.Windows.Controls;
    using System.Windows.Documents;
    using System.Xml;
    
    // (...)
    
    public static class TextBlockHelper
    {
        #region FormattedText Attached dependency property
    
        public static string GetFormattedText(DependencyObject obj)
        {
            return (string)obj.GetValue(FormattedTextProperty);
        }
    
        public static void SetFormattedText(DependencyObject obj, string value)
        {
            obj.SetValue(FormattedTextProperty, value);
        }
    
        public static readonly DependencyProperty FormattedTextProperty =
            DependencyProperty.RegisterAttached("FormattedText",
            typeof(string),
            typeof(TextBlockHelper),
            new UIPropertyMetadata("", FormattedTextChanged));
    
        private static void FormattedTextChanged(DependencyObject sender, DependencyPropertyChangedEventArgs e)
        {
            string value = e.NewValue as string;
    
            TextBlock textBlock = sender as TextBlock;
    
            if (textBlock != null)
            {
                textBlock.Inlines.Clear();
                textBlock.Inlines.Add(Process(value));
            }
        }
    
        #endregion
    
        static Inline Process(string value)
        {
            XmlDocument doc = new XmlDocument();
            doc.LoadXml(value);
    
            Span span = new Span();
            InternalProcess(span, doc.ChildNodes[0]);
    
            return span;
        }
    
        private static void InternalProcess(Span span, XmlNode xmlNode)
        {
            foreach (XmlNode child in xmlNode)
            {
                if (child is XmlText)
                {
                    span.Inlines.Add(new Run(child.InnerText));
                }
                else if (child is XmlElement)
                {
                    Span spanItem = new Span();
                    InternalProcess(spanItem, child);
                    switch (child.Name.ToUpper())
                    {
                        case "B":
                        case "BOLD":
                            Bold bold = new Bold(spanItem);
                            span.Inlines.Add(bold);
                            break;
                        case "I":
                        case "ITALIC":
                            Italic italic = new Italic(spanItem);
                            span.Inlines.Add(italic);
                            break;
                        case "U":
                        case "UNDERLINE":
                            Underline underline = new Underline(spanItem);
                            span.Inlines.Add(underline);
                            break;
                    }
                }
            }
        }
    }
    

    And an example of usage:

    <RootItem xmlns:u="clr-namespace:MyApp.Helpers">
        <TextBlock u:TextBlockHelper.FormattedText="{Binding SomeProperty}" />
    </RootItem>
    
    • 0
    • Reply
    • Share
      Share
      • Share on Facebook
      • Share on Twitter
      • Share on LinkedIn
      • Share on WhatsApp
      • Report

Sidebar

Related Questions

Im having some problems with binding in wpf/xaml. Have this simple file: <Window x:Class=test.Window1
In WPF, is there any way to have the Text property of a TextBlock
I have Window. Window.DataContext = DataRow. and i have TextBlock. I need to bind
I have been trying to reproduce this example: http://www.codeproject.com/KB/WPF/TreeViewWithViewModel.aspx For some reason my treeview
I have the following Window created in WPF. This is how it's shown: I
I have got some WPF source: <Window x:Class=WpfApplication2.MainWindow xmlns=http://schemas.microsoft.com/winfx/2006/xaml/presentation xmlns:x=http://schemas.microsoft.com/winfx/2006/xaml Title=MainWindow Height=350 Width=525> <Window.Resources>
I have a WPF view that displays a Shipment entity. I have a textblock
I have WPF Span that is used as a source to a TextBlock. I
I have the following XAML: <TextBlock Text={Binding ElementName=EditListBox, Path=SelectedItems.Count} Margin=0,0,5,0/> <TextBlock Text=items selected> <TextBlock.Style>
Here's the scenario I have a Grid with some TextBlock controls, each in a

Explore

  • Home
  • Add group
  • Groups page
  • Communities
  • Questions
    • New Questions
    • Trending Questions
    • Must read Questions
    • Hot Questions
  • Polls
  • Tags
  • Badges
  • Users
  • Help
  • SEARCH

Footer

© 2021 The Archive Base. All Rights Reserved
With Love by The Archive Base

Insert/edit link

Enter the destination URL

Or link to existing content

    No search term specified. Showing recent items. Search or use up and down arrow keys to select an item.