Browse Source
git-svn-id: svn://svn.sharpdevelop.net/sharpdevelop/trunk@4207 1ccf3a8d-04fe-1044-b7c0-cef0b8235c61shortcuts
5 changed files with 235 additions and 16 deletions
@ -0,0 +1,176 @@
@@ -0,0 +1,176 @@
|
||||
/* |
||||
* Created by SharpDevelop. |
||||
* User: Daniel |
||||
* Date: 02.06.2009 |
||||
* Time: 21:33 |
||||
* |
||||
* To change this template use Tools | Options | Coding | Edit Standard Headers. |
||||
*/ |
||||
|
||||
using System; |
||||
using System.Collections.Generic; |
||||
using System.Diagnostics; |
||||
using System.Windows; |
||||
using System.Windows.Documents; |
||||
using System.Windows.Media; |
||||
|
||||
namespace ICSharpCode.AvalonEdit.Highlighting |
||||
{ |
||||
/// <summary>
|
||||
/// Takes a series of highlighting commands and stores them.
|
||||
/// Later, it can build inline objects (for use with WPF TextBlock) from the commands.
|
||||
/// </summary>
|
||||
/// <remarks>
|
||||
/// This class is not used in AvalonEdit - but it is useful for someone who wants to put a HighlightedLine
|
||||
/// into a TextBlock.
|
||||
/// In SharpDevelop, we use it to provide syntax highlighting inside the search results pad.
|
||||
/// </remarks>
|
||||
public sealed class HighlightedInlineBuilder |
||||
{ |
||||
sealed class HighlightingState |
||||
{ |
||||
internal Brush Foreground; |
||||
internal FontFamily Family; |
||||
internal FontWeight? Weight; |
||||
internal FontStyle? Style; |
||||
|
||||
public HighlightingState Clone() |
||||
{ |
||||
return new HighlightingState { |
||||
Foreground = this.Foreground, |
||||
Family = this.Family, |
||||
Weight = this.Weight, |
||||
Style = this.Style |
||||
}; |
||||
} |
||||
} |
||||
|
||||
readonly string text; |
||||
List<int> stateChangeOffsets = new List<int>(); |
||||
List<HighlightingState> stateChanges = new List<HighlightingState>(); |
||||
|
||||
int GetIndexForOffset(int offset) |
||||
{ |
||||
if (offset < 0 || offset > text.Length) |
||||
throw new ArgumentOutOfRangeException(); |
||||
int index = stateChangeOffsets.BinarySearch(offset); |
||||
if (index < 0) { |
||||
index = ~index; |
||||
if (offset < text.Length) { |
||||
stateChanges.Insert(index, stateChanges[index - 1].Clone()); |
||||
stateChangeOffsets.Insert(index, offset); |
||||
} |
||||
} |
||||
return index; |
||||
} |
||||
|
||||
/// <summary>
|
||||
/// Creates a new HighlightedInlineBuilder instance.
|
||||
/// </summary>
|
||||
public HighlightedInlineBuilder(string text) |
||||
{ |
||||
if (text == null) |
||||
throw new ArgumentNullException("text"); |
||||
this.text = text; |
||||
stateChangeOffsets.Add(0); |
||||
stateChanges.Add(new HighlightingState()); |
||||
} |
||||
|
||||
/// <summary>
|
||||
/// Gets the text.
|
||||
/// </summary>
|
||||
public string Text { |
||||
get { return text; } |
||||
} |
||||
|
||||
/// <summary>
|
||||
/// Applies the properties from the HighlightingColor to the specified text segment.
|
||||
/// </summary>
|
||||
public void SetHighlighting(int offset, int length, HighlightingColor color) |
||||
{ |
||||
int startIndex = GetIndexForOffset(offset); |
||||
int endIndex = GetIndexForOffset(offset + length); |
||||
for (int i = startIndex; i < endIndex; i++) { |
||||
HighlightingState state = stateChanges[i]; |
||||
if (color.Foreground != null) |
||||
state.Foreground = color.Foreground.GetBrush(null); |
||||
if (color.FontStyle != null) |
||||
state.Style = color.FontStyle; |
||||
if (color.FontWeight != null) |
||||
state.Weight = color.FontWeight; |
||||
} |
||||
} |
||||
|
||||
/// <summary>
|
||||
/// Sets the foreground brush on the specified text segment.
|
||||
/// </summary>
|
||||
public void SetForeground(int offset, int length, Brush brush) |
||||
{ |
||||
int startIndex = GetIndexForOffset(offset); |
||||
int endIndex = GetIndexForOffset(offset + length); |
||||
for (int i = startIndex; i < endIndex; i++) { |
||||
stateChanges[i].Foreground = brush; |
||||
} |
||||
} |
||||
|
||||
/// <summary>
|
||||
/// Sets the font weight on the specified text segment.
|
||||
/// </summary>
|
||||
public void SetFontWeight(int offset, int length, FontWeight weight) |
||||
{ |
||||
int startIndex = GetIndexForOffset(offset); |
||||
int endIndex = GetIndexForOffset(offset + length); |
||||
for (int i = startIndex; i < endIndex; i++) { |
||||
stateChanges[i].Weight = weight; |
||||
} |
||||
} |
||||
|
||||
/// <summary>
|
||||
/// Sets the font style on the specified text segment.
|
||||
/// </summary>
|
||||
public void SetFontStyle(int offset, int length, FontStyle style) |
||||
{ |
||||
int startIndex = GetIndexForOffset(offset); |
||||
int endIndex = GetIndexForOffset(offset + length); |
||||
for (int i = startIndex; i < endIndex; i++) { |
||||
stateChanges[i].Style = style; |
||||
} |
||||
} |
||||
|
||||
/// <summary>
|
||||
/// Sets the font family on the specified text segment.
|
||||
/// </summary>
|
||||
public void SetFontFamily(int offset, int length, FontFamily family) |
||||
{ |
||||
int startIndex = GetIndexForOffset(offset); |
||||
int endIndex = GetIndexForOffset(offset + length); |
||||
for (int i = startIndex; i < endIndex; i++) { |
||||
stateChanges[i].Family = family; |
||||
} |
||||
} |
||||
|
||||
/// <summary>
|
||||
/// Creates WPF Run instances that can be used for TextBlock.Inlines.
|
||||
/// </summary>
|
||||
public Run[] CreateInlines() |
||||
{ |
||||
Run[] runs = new Run[stateChanges.Count]; |
||||
for (int i = 0; i < runs.Length; i++) { |
||||
int startOffset = stateChangeOffsets[i]; |
||||
int endOffset = i + 1 < stateChangeOffsets.Count ? stateChangeOffsets[i + 1] : text.Length; |
||||
Run r = new Run(text.Substring(startOffset, endOffset - startOffset)); |
||||
HighlightingState state = stateChanges[i]; |
||||
if (state.Foreground != null) |
||||
r.Foreground = state.Foreground; |
||||
if (state.Weight != null) |
||||
r.FontWeight = state.Weight.Value; |
||||
if (state.Family != null) |
||||
r.FontFamily = state.Family; |
||||
if (state.Style != null) |
||||
r.FontStyle = state.Style.Value; |
||||
runs[i] = r; |
||||
} |
||||
return runs; |
||||
} |
||||
} |
||||
} |
||||
Loading…
Reference in new issue