Добавить гиперссылку на текстовый блок wpf - PullRequest
32 голосов
/ 19 января 2010

Привет, У меня есть текст в БД, и это выглядит следующим образом:

Lorem Ipsum Dolor Sit Amet, Concetetur Adipiscing Elit. DUIs Tellus NISL, Venenatis и Pharetra AC, Tempor Sed Sapien. целое число pellentesque blandit велит, в виске урна semper сидеть амет. DUIs моллис, либеро ут констетур интердум, масса теллс посуре ниси, ес аликет элит лакус некрат. Представляем товарный квам. ** [а href = 'http://somesite.com']some site [/ a] ** Suspendisse at nisi sit amet Mass Molestie Gravida Feugiat AC сем. Phasellus ac mauris ipsum, вел auctor odio

Мой вопрос: как я могу отобразить Hyperlink в TextBlock? Я не хочу использовать элемент управления webBrowser для этой цели. Я также не хочу использовать этот элемент управления: http://www.codeproject.com/KB/WPF/htmltextblock.aspx также

Ответы [ 3 ]

90 голосов
/ 19 января 2010

Отображение довольно простое, навигация это другой вопрос. XAML выглядит так:

<TextBlock Name="TextBlockWithHyperlink">
    Some text 
    <Hyperlink 
        NavigateUri="http://somesite.com"
        RequestNavigate="Hyperlink_RequestNavigate">
        some site
    </Hyperlink>
    some more text
</TextBlock>

И обработчик событий, который запускает браузер по умолчанию для перехода к вашей гиперссылке, будет:

private void Hyperlink_RequestNavigate(object sender, RequestNavigateEventArgs e) {
    System.Diagnostics.Process.Start(e.Uri.ToString());
}

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

TextBlockWithHyperlink.Inlines.Clear();
TextBlockWithHyperlink.Inlines.Add("Some text ");
Hyperlink hyperLink = new Hyperlink() {
    NavigateUri = new Uri("http://somesite.com")
};
hyperLink.Inlines.Add("some site");
hyperLink.RequestNavigate += Hyperlink_RequestNavigate;
TextBlockWithHyperlink.Inlines.Add(hyperLink);
TextBlockWithHyperlink.Inlines.Add(" Some more text");
15 голосов
/ 22 января 2010

Вы можете использовать Regex с преобразователем значений в такой ситуации.

Используйте это для ваших требований (оригинальная идея от здесь ):

    private Regex regex = 
        new Regex(@"\[a\s+href='(?<link>[^']+)'\](?<text>.*?)\[/a\]",
        RegexOptions.Compiled);

Это будет соответствовать всем ссылкам в вашей строке, содержащей ссылки, и создавать 2 именованные группы для каждого соответствия: link и text

Теперь вы можете перебирать все совпадения. Каждый матч даст вам

    foreach (Match match in regex.Matches(stringContainingLinks))
    { 
        string link    = match.Groups["link"].Value;
        int link_start = match.Groups["link"].Index;
        int link_end   = match.Groups["link"].Index + link.Length;

        string text    = match.Groups["text"].Value;
        int text_start = match.Groups["text"].Index;
        int text_end   = match.Groups["text"].Index + text.Length;

        // do whatever you want with stringContainingLinks.
        // In particular, remove whole `match` ie [a href='...']...[/a]
        // and instead put HyperLink with `NavigateUri = link` and
        // `Inlines.Add(text)` 
        // See the answer by Stanislav Kniazev for how to do this
    }

Примечание: используйте эту логику в своем пользовательском ConvertToHyperlinkedText преобразователе значений.

0 голосов
/ 20 апреля 2019

Еще одна версия этого и не полностью совпадающая с распознаванием формата здесь, но вот класс для автоматического распознавания ссылок в фрагменте текста и превращения их в живые гиперссылки:

internal class TextBlockExt
{
    static Regex _regex =
        new Regex(@"http[s]?://[^\s-]+",
                  RegexOptions.Compiled);

    public static readonly DependencyProperty FormattedTextProperty = DependencyProperty.RegisterAttached("FormattedText", 
        typeof(string), typeof(TextBlockExt), new FrameworkPropertyMetadata(string.Empty, FrameworkPropertyMetadataOptions.AffectsMeasure, FormattedTextPropertyChanged));
    public static void SetFormattedText(DependencyObject textBlock, string value)
    { textBlock.SetValue(FormattedTextProperty, value); }

    public static string GetFormattedText(DependencyObject textBlock)
    { return (string)textBlock.GetValue(FormattedTextProperty); }

    static void FormattedTextPropertyChanged(DependencyObject d, DependencyPropertyChangedEventArgs e)
    {
        if (!(d is TextBlock textBlock)) return; 

        var formattedText = (string)e.NewValue ?? string.Empty;
        string fullText =
            $"<Span xml:space=\"preserve\" xmlns=\"http://schemas.microsoft.com/winfx/2006/xaml/presentation\">{formattedText}</Span>";

        textBlock.Inlines.Clear();
        using (var xmlReader1 = XmlReader.Create(new StringReader(fullText)))
        {
            try
            {
                var result = (Span)XamlReader.Load(xmlReader1);
                RecognizeHyperlinks(result);
                textBlock.Inlines.Add(result);
            }
            catch
            {
                formattedText = System.Security.SecurityElement.Escape(formattedText);
                fullText =
                    $"<Span xml:space=\"preserve\" xmlns=\"http://schemas.microsoft.com/winfx/2006/xaml/presentation\">{formattedText}</Span>";

                using (var xmlReader2 = XmlReader.Create(new StringReader(fullText)))
                {
                    try
                    {
                        dynamic result = (Span) XamlReader.Load(xmlReader2);
                        textBlock.Inlines.Add(result);
                    }
                    catch
                    {
                        //ignored
                    }
                }
            }
        }
    }

    static void RecognizeHyperlinks(Inline originalInline)
    {
        if (!(originalInline is Span span)) return;

        var replacements = new Dictionary<Inline, List<Inline>>();
        var startInlines = new List<Inline>(span.Inlines);
        foreach (Inline i in startInlines)
        {
            switch (i)
            {
                case Hyperlink _:
                    continue;
                case Run run:
                {
                    if (!_regex.IsMatch(run.Text)) continue;
                    var newLines = GetHyperlinks(run);
                    replacements.Add(run, newLines);
                    break;
                }
                default:
                    RecognizeHyperlinks(i);
                    break;
            }
        }

        if (!replacements.Any()) return;

        var currentInlines = new List<Inline>(span.Inlines);
        span.Inlines.Clear();
        foreach (Inline i in currentInlines)
        {
            if (replacements.ContainsKey(i)) span.Inlines.AddRange(replacements[i]);
            else span.Inlines.Add(i);
        }
    }

    static List<Inline> GetHyperlinks(Run run)
    {
        var result = new List<Inline>();
        var currentText = run.Text;
        do
        {
            if (!_regex.IsMatch(currentText))
            {
                if (!string.IsNullOrEmpty(currentText)) result.Add(new Run(currentText));
                break;
            }
            var match = _regex.Match(currentText);

            if (match.Index > 0)
            {
                result.Add(new Run(currentText.Substring(0, match.Index)));
            }

            var hyperLink = new Hyperlink() { NavigateUri = new Uri(match.Value) };
            hyperLink.Inlines.Add(match.Value);
            hyperLink.RequestNavigate += HyperLink_RequestNavigate;
            result.Add(hyperLink);

            currentText = currentText.Substring(match.Index + match.Length);
        } while (true);

        return result;
    }

    static void HyperLink_RequestNavigate(object sender, System.Windows.Navigation.RequestNavigateEventArgs e)
    {
        try
        {
            Process.Start(e.Uri.ToString());
        }
        catch { }
    }
}

Используя это, вы можете просто сделать <TextBlock ns:Attached.FormattedText="{Binding Content}" /> вместо <TextBlock Text="{Binding Content}" />, и он автоматически распознает и активирует ссылки, а также распознает обычные теги форматирования, такие как <Bold> и т. Д.

Обратите внимание, что это основано на ответе @gwiazdorrr здесь , а также на некоторых других ответах на этот вопрос; Я в основном объединил их все в 1 и сделал некоторую обработку рекурсии, и это работает! :). Шаблоны и системы также могут быть адаптированы для распознавания других типов ссылок или разметки, если это необходимо.

...