Добавить гиперссылку на текстовый блок wpf
Привет,
У меня есть текст в db, и он выглядит следующим образом:
Lorem ipsum dolor sit amet, consectetur adipiscing elit. DUIs tellus nisl, venenatis et pharetra ac, tempor sed sapien. целое число pellentesque blandit velit, в tempus urna semper sit amet. DUIs mollis, libero ut consectetur interdum, massa tellus posuere nisi, eu aliquet elit lacus nec erat. Представьте коммо-кэм. ** [а href= 'http://somesite.com'] некоторый сайт [/a] ** Suspendisse at nisi sit amet massa molestie gravida feugiat ac sem. Phasellus ac mauris ipsum, vel auctor odio
Мой вопрос: как я могу отобразить Hyperlink
в TextBlock
? Для этой цели я не хочу использовать элемент управления webBrowser.
Я также не хочу использовать этот элемент управления: http://www.codeproject.com/KB/WPF/htmltextblock.aspx также
Ответы
Ответ 1
В такой ситуации вы можете использовать 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
.
Ответ 2
Отображение довольно просто, навигация - это еще один вопрос. 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");
Ответ 3
Другая версия этого и не совсем то же самое, что и распознавание формата здесь, но вот класс для автоматического распознавания ссылок в фрагменте текста и превращения их в живые гиперссылки:
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 и сделал некоторую обработку рекурсии, и это работает! :). Шаблоны и системы также могут быть адаптированы для распознавания других типов ссылок или разметки, если это необходимо.