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 @@ |
|||||||
|
/* |
||||||
|
* 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