|
// Licensed to the .NET Foundation under one or more agreements.
// The .NET Foundation licenses this file to you under the MIT license.
// See the LICENSE file in the project root for more information.
using System.Globalization;
using System.Windows;
using System.Windows.Documents;
using System.Windows.Media;
using System.Windows.Media.TextFormatting;
using MS.Internal.Text;
using MS.Internal.Documents;
using MS.Internal.PtsHost.UnsafeNativeMethods;
namespace MS.Internal.PtsHost
{
/// <summary>
/// Text line formatter.
/// </summary>
/// <remarks>
/// NOTE: All DCPs used during line formatting are related to cpPara.
/// To get abosolute CP, add cpPara to a dcp value.
/// </remarks>
internal sealed class Line : LineBase
{
//-------------------------------------------------------------------
//
// Constructors
//
//-------------------------------------------------------------------
#region Constructors
/// <summary>
/// Constructor.
/// </summary>
/// <param name="host">
/// TextFormatter host
/// </param>
/// <param name="paraClient">
/// Owner of the line
/// </param>
/// <param name="cpPara">
/// CP of the beginning of the text paragraph
/// </param>
internal Line(TextFormatterHost host, TextParaClient paraClient, int cpPara) : base(paraClient)
{
_host = host;
_cpPara = cpPara;
_textAlignment = (TextAlignment)TextParagraph.Element.GetValue(Block.TextAlignmentProperty);
_indent = 0.0;
}
/// <summary>
/// Free all resources associated with the line. Prepare it for reuse.
/// </summary>
public override void Dispose()
{
Debug.Assert(_line != null, "Line has been already disposed.");
try
{
if (_line != null)
{
_line.Dispose();
}
}
finally
{
_line = null;
_runs = null;
_hasFigures = false;
_hasFloaters = false;
base.Dispose();
}
}
#endregion Constructors
// ------------------------------------------------------------------
//
// PTS Callbacks
//
// ------------------------------------------------------------------
#region PTS Callbacks
/// <summary>
/// GetDvrSuppressibleBottomSpace
/// </summary>
/// <param name="dvrSuppressible">
/// OUT: empty space suppressible at the bottom
/// </param>
internal void GetDvrSuppressibleBottomSpace(
out int dvrSuppressible)
{
dvrSuppressible = Math.Max(0, TextDpi.ToTextDpi(_line.OverhangAfter));
}
/// <summary>
/// GetDurFigureAnchor
/// </summary>
/// <param name="paraFigure">
/// IN: FigureParagraph for which we require anchor dur
/// </param>
/// <param name="fswdir">
/// IN: current direction
/// </param>
/// <param name="dur">
/// OUT: distance from the beginning of the line to the anchor
/// </param>
internal void GetDurFigureAnchor(
FigureParagraph paraFigure,
uint fswdir,
out int dur)
{
int cpFigure = TextContainerHelper.GetCPFromElement(_paraClient.Paragraph.StructuralCache.TextContainer, paraFigure.Element, ElementEdge.BeforeStart);
int dcpFigure = cpFigure - _cpPara;
double distance = _line.GetDistanceFromCharacterHit(new CharacterHit(dcpFigure, 0));
dur = TextDpi.ToTextDpi(distance);
}
#endregion PTS Callbacks
// ------------------------------------------------------------------
//
// TextSource Implementation
//
// ------------------------------------------------------------------
#region TextSource Implementation
/// <summary>
/// Get a text run at specified text source position and return it.
/// </summary>
/// <param name="dcp">
/// dcp of position relative to start of line
/// </param>
internal override TextRun GetTextRun(int dcp)
{
TextRun run = null;
ITextContainer textContainer = _paraClient.Paragraph.StructuralCache.TextContainer;
StaticTextPointer position = textContainer.CreateStaticPointerAtOffset(_cpPara + dcp);
switch (position.GetPointerContext(LogicalDirection.Forward))
{
case TextPointerContext.Text:
run = HandleText(position);
break;
case TextPointerContext.ElementStart:
run = HandleElementStartEdge(position);
break;
case TextPointerContext.ElementEnd:
run = HandleElementEndEdge(position);
break;
case TextPointerContext.EmbeddedElement:
run = HandleEmbeddedObject(dcp, position);
break;
case TextPointerContext.None:
run = new ParagraphBreakRun(_syntheticCharacterLength, PTS.FSFLRES.fsflrEndOfParagraph);
break;
}
Invariant.Assert(run != null, "TextRun has not been created.");
Invariant.Assert(run.Length > 0, "TextRun has to have positive length.");
return run;
}
/// <summary>
/// Get text immediately before specified text source position. Return CharacterBufferRange
/// containing this text.
/// </summary>
/// <param name="dcp">
/// dcp of position relative to start of line
/// </param>
internal override TextSpan<CultureSpecificCharacterBufferRange> GetPrecedingText(int dcp)
{
// Parameter validation
Invariant.Assert(dcp >= 0);
int nonTextLength = 0;
CharacterBufferRange precedingText = CharacterBufferRange.Empty;
CultureInfo culture = null;
if (dcp > 0)
{
// Create TextPointer at dcp, and pointer at paragraph start to compare
ITextPointer startPosition = TextContainerHelper.GetTextPointerFromCP(_paraClient.Paragraph.StructuralCache.TextContainer, _cpPara, LogicalDirection.Forward);
ITextPointer position = TextContainerHelper.GetTextPointerFromCP(_paraClient.Paragraph.StructuralCache.TextContainer, _cpPara + dcp, LogicalDirection.Forward);
// Move backward until we find a position at the end of a text run, or reach start of TextContainer
while (position.GetPointerContext(LogicalDirection.Backward) != TextPointerContext.Text &&
position.CompareTo(startPosition) != 0)
{
position.MoveByOffset(-1);
nonTextLength++;
}
// Return text in run. If it is at start of TextContainer this will return an empty string
string precedingTextString = position.GetTextInRun(LogicalDirection.Backward);
precedingText = new CharacterBufferRange(precedingTextString, 0, precedingTextString.Length);
StaticTextPointer pointer = position.CreateStaticPointer();
DependencyObject element = pointer.Parent ?? _paraClient.Paragraph.Element;
culture = DynamicPropertyReader.GetCultureInfo(element);
}
return new TextSpan<CultureSpecificCharacterBufferRange>(
nonTextLength + precedingText.Length,
new CultureSpecificCharacterBufferRange(culture, precedingText)
);
}
/// <summary>
/// Get Text effect index from text source character index. Return int value of Text effect index.
/// </summary>
/// <param name="dcp">
/// dcp of CharacterHit relative to start of line
/// </param>
internal override int GetTextEffectCharacterIndexFromTextSourceCharacterIndex(int dcp)
{
return _cpPara + dcp;
}
#endregion TextSource Implementation
// ------------------------------------------------------------------
//
// Internal Methods
//
// ------------------------------------------------------------------
#region Internal Methods
/// <summary>
/// Create and format text line.
/// </summary>
/// <param name="ctx">
/// Line formatting context.
/// </param>
/// <param name="dcp">
/// Character position where the line starts.
/// </param>
/// <param name="width">
/// Requested width of the line.
/// </param>
/// <param name="trackWidth">
/// Requested width of track.
/// </param>
/// <param name="lineProps">
/// Line properties.
/// </param>
/// <param name="textLineBreak">
/// Line break object.
/// </param>
internal void Format(FormattingContext ctx, int dcp, int width, int trackWidth, TextParagraphProperties lineProps, TextLineBreak textLineBreak)
{
// Set formatting context
_formattingContext = ctx;
_dcp = dcp;
_host.Context = this;
_wrappingWidth = TextDpi.FromTextDpi(width);
_trackWidth = TextDpi.FromTextDpi(trackWidth);
_mirror = (lineProps.FlowDirection == FlowDirection.RightToLeft);
_indent = lineProps.Indent;
try
{
// Create line object
if(ctx.LineFormatLengthTarget == -1)
{
_line = _host.TextFormatter.FormatLine(_host, dcp, _wrappingWidth, lineProps, textLineBreak, ctx.TextRunCache);
}
else
{
_line = _host.TextFormatter.RecreateLine(_host, dcp, ctx.LineFormatLengthTarget, _wrappingWidth, lineProps, textLineBreak, ctx.TextRunCache);
}
_runs = _line.GetTextRunSpans();
Invariant.Assert(_runs != null, "Cannot retrieve runs collection.");
// Submit inline objects (only in measure mode)
if (_formattingContext.MeasureMode)
{
List<InlineObject> inlineObjects = new List<InlineObject>(1);
int dcpRun = _dcp;
// Enumerate through all runs in the current line and retrieve
// all inline objects.
// If there are any figures / floaters, store this information for later use.
foreach (TextSpan<TextRun> textSpan in _runs)
{
TextRun run = (TextRun)textSpan.Value;
if (run is InlineObjectRun)
{
inlineObjects.Add(new InlineObject(dcpRun, ((InlineObjectRun)run).UIElementIsland, (TextParagraph)_paraClient.Paragraph));
}
else if (run is FloatingRun)
{
if (((FloatingRun)run).Figure)
{
_hasFigures = true;
}
else
{
_hasFloaters = true;
}
}
// Do not use TextRun.Length, because it gives total length of the run.
// So, if the run is broken between lines, it gives incorrect value.
// Use length of the TextSpan instead, which gives the correct length here.
dcpRun += textSpan.Length;
}
// Submit inline objects to the paragraph cache
if (inlineObjects.Count == 0)
{
inlineObjects = null;
}
TextParagraph.SubmitInlineObjects(dcp, dcp + ActualLength, inlineObjects);
}
}
finally
{
// Clear formatting context
_host.Context = null;
}
}
/// <summary>
/// Measure child UIElement.
/// </summary>
/// <param name="inlineObject">
/// Element whose size we are measuring
/// </param>
/// <returns>
/// Size of the child UIElement
/// </returns>
internal Size MeasureChild(InlineObjectRun inlineObject)
{
// Measure inline object only during measure pass. Otherwise
// use cached data.
Size desiredSize;
if (_formattingContext.MeasureMode)
{
Debug.Assert(!double.IsNaN(_trackWidth), "Track width must be set for measure pass.");
// Always measure at infinity for bottomless, consistent constraint.
double pageHeight = _paraClient.Paragraph.StructuralCache.CurrentFormatContext.DocumentPageSize.Height;
if (!_paraClient.Paragraph.StructuralCache.CurrentFormatContext.FinitePage)
{
pageHeight = Double.PositiveInfinity;
}
desiredSize = inlineObject.UIElementIsland.DoLayout(new Size(_trackWidth, pageHeight), true, true);
}
else
{
desiredSize = inlineObject.UIElementIsland.Root.DesiredSize;
}
return desiredSize;
}
/// <summary>
/// Create and return visual node for the line.
/// </summary>
internal ContainerVisual CreateVisual()
{
LineVisual visual = new LineVisual();
// Set up the text source for rendering callback
_host.Context = this;
try
{
// Handle text trimming.
IList<TextSpan<TextRun>> runs = _runs;
System.Windows.Media.TextFormatting.TextLine line = _line;
if (_line.HasOverflowed && TextParagraph.Properties.TextTrimming != TextTrimming.None)
{
line = _line.Collapse(GetCollapsingProps(_wrappingWidth, TextParagraph.Properties));
Invariant.Assert(line.HasCollapsed, "Line has not been collapsed");
runs = line.GetTextRunSpans();
}
// Add visuals for all embedded elements.
if (HasInlineObjects())
{
VisualCollection visualChildren = visual.Children;
// Get flow direction of the paragraph element.
DependencyObject paragraphElement = _paraClient.Paragraph.Element;
FlowDirection paragraphFlowDirection = (FlowDirection)paragraphElement.GetValue(FrameworkElement.FlowDirectionProperty);
// Before text rendering, add all visuals for inline objects.
int dcpRun = _dcp;
// Enumerate through all runs in the current line and connect visuals for all inline objects.
foreach (TextSpan<TextRun> textSpan in runs)
{
TextRun run = (TextRun)textSpan.Value;
if (run is InlineObjectRun inlineObject)
{
FlowDirection flowDirection;
Rect rect = GetBoundsFromPosition(dcpRun, run.Length, out flowDirection);
Debug.Assert(DoubleUtil.GreaterThanOrClose(rect.Width, 0), "Negative inline object's width.");
// Disconnect visual from its old parent, if necessary.
Visual currentParent = VisualTreeHelper.GetParent(inlineObject.UIElementIsland) as Visual;
if (currentParent != null)
{
ContainerVisual parent = currentParent as ContainerVisual;
Invariant.Assert(parent != null, "Parent should always derives from ContainerVisual.");
parent.Children.Remove(inlineObject.UIElementIsland);
}
if (!line.HasCollapsed || ((rect.Left + inlineObject.UIElementIsland.Root.DesiredSize.Width) < line.Width))
{
// Check parent's FlowDirection to determine if mirroring is needed
if (inlineObject.UIElementIsland.Root is FrameworkElement)
{
DependencyObject parent = ((FrameworkElement)inlineObject.UIElementIsland.Root).Parent;
FlowDirection parentFlowDirection = (FlowDirection)parent.GetValue(FrameworkElement.FlowDirectionProperty);
PtsHelper.UpdateMirroringTransform(paragraphFlowDirection, parentFlowDirection, inlineObject.UIElementIsland, rect.Width);
}
visualChildren.Add(inlineObject.UIElementIsland);
inlineObject.UIElementIsland.Offset = new Vector(rect.Left, rect.Top);
}
}
// Do not use TextRun.Length, because it gives total length of the run.
// So, if the run is broken between lines, it gives incorrect value.
// Use length of the TextSpan instead, which gives the correct length here.
dcpRun += textSpan.Length;
}
}
// Calculate shift in line offset to render trailing spaces or avoid clipping text
double delta = TextDpi.FromTextDpi(CalculateUOffsetShift());
DrawingContext ctx = visual.Open();
line.Draw(ctx, new Point(delta, 0), (_mirror ? InvertAxes.Horizontal : InvertAxes.None));
ctx.Close();
visual.WidthIncludingTrailingWhitespace = line.WidthIncludingTrailingWhitespace - _indent;
}
finally
{
_host.Context = null; // clear the context
}
return visual;
}
/// <summary>
/// Return bounds of an object/character at specified text position.
/// </summary>
/// <param name="textPosition">
/// Position of the object/character
/// </param>
/// <param name="flowDirection">
/// Flow direction of the object/character
/// </param>
internal Rect GetBoundsFromTextPosition(int textPosition, out FlowDirection flowDirection)
{
return GetBoundsFromPosition(textPosition, 1, out flowDirection);
}
/// <summary>
/// Returns an ArrayList of rectangles (Rect) that form the bounds of the region specified between
/// the start and end points
/// </summary>
/// <param name="cp"></param>
/// int offset indicating the starting point of the region for which bounds are required
/// <param name="cch">
/// Length in characters of the region for which bounds are required
/// </param>
/// <param name="xOffset">
/// Offset of line in x direction, to be added to line bounds to get actual rectangle for line
/// </param>
/// <param name="yOffset">
/// Offset of line in y direction, to be added to line bounds to get actual rectangle for line
/// </param>
/// <remarks>
/// This function calls GetTextBounds for the line, and then checks if there are text run bounds. If they exist,
/// it uses those as the bounding rectangles. If not, it returns the rectangle for the first (and only) element
/// of the text bounds.
/// </remarks>
internal List<Rect> GetRangeBounds(int cp, int cch, double xOffset, double yOffset)
{
List<Rect> rectangles = new List<Rect>();
// Calculate shift in line offset to render trailing spaces or avoid clipping text
double delta = TextDpi.FromTextDpi(CalculateUOffsetShift());
double newUOffset = xOffset + delta;
IList<TextBounds> textBounds;
if (_line.HasOverflowed && TextParagraph.Properties.TextTrimming != TextTrimming.None)
{
// Verify that offset shift is 0 for this case. We should never shift offsets when ellipses are
// rendered.
Invariant.Assert(DoubleUtil.AreClose(delta, 0));
System.Windows.Media.TextFormatting.TextLine line = _line.Collapse(GetCollapsingProps(_wrappingWidth, TextParagraph.Properties));
Invariant.Assert(line.HasCollapsed, "Line has not been collapsed");
textBounds = line.GetTextBounds(cp, cch);
}
else
{
textBounds = _line.GetTextBounds(cp, cch);
}
Invariant.Assert(textBounds.Count > 0);
for (int boundIndex = 0; boundIndex < textBounds.Count; boundIndex++)
{
Rect rect = textBounds[boundIndex].Rectangle;
rect.X += newUOffset;
rect.Y += yOffset;
rectangles.Add(rect);
}
return rectangles;
}
/// <summary>
/// Passes line break object out from underlying line object
/// </summary>
internal TextLineBreak GetTextLineBreak()
{
if(_line == null)
{
return null;
}
return _line.GetTextLineBreak();
}
/// <summary>
/// Return text position index from the given distance.
/// </summary>
/// <param name="urDistance">
/// Distance relative to the beginning of the line.
/// </param>
internal CharacterHit GetTextPositionFromDistance(int urDistance)
{
// Calculate shift in line offset to render trailing spaces or avoid clipping text
int delta = CalculateUOffsetShift();
if (_line.HasOverflowed && TextParagraph.Properties.TextTrimming != TextTrimming.None)
{
System.Windows.Media.TextFormatting.TextLine line = _line.Collapse(GetCollapsingProps(_wrappingWidth, TextParagraph.Properties));
Invariant.Assert(delta == 0);
Invariant.Assert(line.HasCollapsed, "Line has not been collapsed");
return line.GetCharacterHitFromDistance(TextDpi.FromTextDpi(urDistance));
}
return _line.GetCharacterHitFromDistance(TextDpi.FromTextDpi(urDistance - delta));
}
/// <summary>
/// Hit tests to the correct ContentElement within the line.
/// </summary>
/// <param name="urOffset">
/// Offset within the line.
/// </param>
/// <returns>
/// ContentElement which has been hit.
/// </returns>
internal IInputElement InputHitTest(int urOffset)
{
DependencyObject element = null;
TextPointer position;
TextPointerContext type = TextPointerContext.None;
CharacterHit charIndex;
int cp, delta;
// Calculate shift in line offset to render trailing spaces or avoid clipping text
delta = CalculateUOffsetShift();
if (_line.HasOverflowed && TextParagraph.Properties.TextTrimming != TextTrimming.None)
{
// We should not shift offset in this case
Invariant.Assert(delta == 0);
System.Windows.Media.TextFormatting.TextLine line = _line.Collapse(GetCollapsingProps(_wrappingWidth, TextParagraph.Properties));
Invariant.Assert(line.HasCollapsed, "Line has not been collapsed");
// Get TextPointer from specified distance.
charIndex = line.GetCharacterHitFromDistance(TextDpi.FromTextDpi(urOffset));
}
else
{
// Get TextPointer from specified distance.
charIndex = _line.GetCharacterHitFromDistance(TextDpi.FromTextDpi(urOffset - delta));
}
cp = _paraClient.Paragraph.ParagraphStartCharacterPosition + charIndex.FirstCharacterIndex + charIndex.TrailingLength;
position = TextContainerHelper.GetTextPointerFromCP(_paraClient.Paragraph.StructuralCache.TextContainer, cp, LogicalDirection.Forward) as TextPointer;
if (position != null)
{
// If start of character, look forward. Otherwise, look backward.
type = position.GetPointerContext((charIndex.TrailingLength == 0) ? LogicalDirection.Forward : LogicalDirection.Backward);
// Get element only for Text & Start/End element, for all other positions
// return null (it means that the line owner has been hit).
if (type == TextPointerContext.Text || type == TextPointerContext.ElementEnd)
{
element = position.Parent;
}
else if (type == TextPointerContext.ElementStart)
{
element = position.GetAdjacentElementFromOuterPosition(LogicalDirection.Forward);
}
}
return element as IInputElement;
}
/// <summary>
/// Get length of content hidden by ellipses. Return integer length of this content.
/// </summary>
internal int GetEllipsesLength()
{
// There are no ellipses, if:
// * there is no overflow in the line
// * text trimming is turned off
if (!_line.HasOverflowed)
{
return 0;
}
if (TextParagraph.Properties.TextTrimming == TextTrimming.None)
{
return 0;
}
// Create collapsed text line to get length of collapsed content.
System.Windows.Media.TextFormatting.TextLine collapsedLine = _line.Collapse(GetCollapsingProps(_wrappingWidth, TextParagraph.Properties));
Invariant.Assert(collapsedLine.HasCollapsed, "Line has not been collapsed");
IList<TextCollapsedRange> collapsedRanges = collapsedLine.GetTextCollapsedRanges();
if (collapsedRanges != null)
{
Invariant.Assert(collapsedRanges.Count == 1, "Multiple collapsed ranges are not supported.");
TextCollapsedRange collapsedRange = collapsedRanges[0];
return collapsedRange.Length;
}
return 0;
}
/// <summary>
/// Retrieves collection of GlyphRuns from a range of text.
/// </summary>
/// <param name="glyphRuns">
/// Glyph runs.
/// </param>
/// <param name="dcpStart">
/// Start dcp of range
/// </param>
/// <param name="dcpEnd">
/// End dcp of range.
/// </param>
internal void GetGlyphRuns(List<GlyphRun> glyphRuns, int dcpStart, int dcpEnd)
{
// NOTE: Following logic is only temporary workaround for lack
// of appropriate API that should be exposed by TextLine.
int dcp = dcpStart - _dcp;
int cch = dcpEnd - dcpStart;
Debug.Assert(dcp >= 0 && (dcp + cch <= _line.Length));
IList<TextSpan<TextRun>> spans = _line.GetTextRunSpans();
DrawingGroup drawing = new DrawingGroup();
DrawingContext ctx = drawing.Open();
// Calculate shift in line offset to render trailing spaces or avoid clipping text
double delta = TextDpi.FromTextDpi(CalculateUOffsetShift());
_line.Draw(ctx, new Point(delta, 0), InvertAxes.None);
ctx.Close();
// Copy glyph runs into separate array (for backward navigation).
// And count number of chracters in the glyph runs collection.
int cchGlyphRuns = 0;
List<GlyphRun> glyphRunsCollection = new(4);
AddGlyphRunRecursive(drawing, glyphRunsCollection, ref cchGlyphRuns);
Debug.Assert(cchGlyphRuns > 0 && glyphRunsCollection.Count > 0);
// Count number of characters in text runs.
int cchTextSpans = 0;
foreach (TextSpan<TextRun> textSpan in spans)
{
if (textSpan.Value is TextCharacters)
{
cchTextSpans += textSpan.Length;
}
}
// If number of characters in glyph runs is greater than number of characters
// in text runs, it means that there is bullet at the beginning of the line
// or hyphen at the end of the line.
// For now hyphen case is ignored.
// Remove those glyph runs from our colleciton.
while (cchGlyphRuns > cchTextSpans)
{
GlyphRun glyphRun = glyphRunsCollection[0];
cchGlyphRuns -= (glyphRun.Characters == null ? 0 : glyphRun.Characters.Count);
glyphRunsCollection.RemoveAt(0);
}
int curDcp = 0;
int runIndex = 0;
foreach (TextSpan<TextRun> span in spans)
{
if (span.Value is TextCharacters)
{
int cchRunsInSpan = 0;
while (cchRunsInSpan < span.Length)
{
Invariant.Assert(runIndex < glyphRunsCollection.Count);
GlyphRun run = glyphRunsCollection[runIndex];
int characterCount = (run.Characters == null ? 0 : run.Characters.Count);
if ((dcp < curDcp + characterCount) && (dcp + cch > curDcp))
{
glyphRuns.Add(run);
}
cchRunsInSpan += characterCount;
++runIndex;
}
Invariant.Assert(cchRunsInSpan == span.Length);
// No need to continue, if dcpEnd has been reached.
if (dcp + cch <= curDcp + span.Length)
break;
}
curDcp += span.Length;
}
}
/// <summary>
/// Return text position for next caret position
/// </summary>
/// <param name="index">
/// CharacterHit for current position
/// </param>
internal CharacterHit GetNextCaretCharacterHit(CharacterHit index)
{
return _line.GetNextCaretCharacterHit(index);
}
/// <summary>
/// Return text position for previous caret position
/// </summary>
/// <param name="index">
/// CharacterHit for current position
/// </param>
internal CharacterHit GetPreviousCaretCharacterHit(CharacterHit index)
{
return _line.GetPreviousCaretCharacterHit(index);
}
/// <summary>
/// Return text position for backspace caret position
/// </summary>
/// <param name="index">
/// CharacterHit for current position
/// </param>
internal CharacterHit GetBackspaceCaretCharacterHit(CharacterHit index)
{
return _line.GetBackspaceCaretCharacterHit(index);
}
/// <summary>
/// Returns true of char hit is at caret unit boundary.
/// </summary>
/// <param name="charHit">
/// CharacterHit to be tested.
/// </param>
internal bool IsAtCaretCharacterHit(CharacterHit charHit)
{
return _line.IsAtCaretCharacterHit(charHit, _dcp);
}
#endregion Internal Methods
//-------------------------------------------------------------------
//
// Internal Properties
//
//-------------------------------------------------------------------
#region Internal Properties
/// <summary>
/// Distance from the beginning of paragraph edge to the line edge.
/// </summary>
internal int Start
{
get
{
return TextDpi.ToTextDpi(_line.Start) + TextDpi.ToTextDpi(_indent) + CalculateUOffsetShift();
}
}
/// <summary>
/// Calculated width of the line.
/// </summary>
internal int Width
{
get
{
int width;
if (IsWidthAdjusted)
{
width = TextDpi.ToTextDpi(_line.WidthIncludingTrailingWhitespace) - TextDpi.ToTextDpi(_indent);
}
else
{
width = TextDpi.ToTextDpi(_line.Width) - TextDpi.ToTextDpi(_indent);
}
Invariant.Assert(width >= 0, "Line width cannot be negative");
return width;
}
}
/// <summary>
/// Height of the line; line advance distance.
/// </summary>
internal int Height
{
get
{
return TextDpi.ToTextDpi(_line.Height);
}
}
/// <summary>
/// Baseline offset from the top of the line.
/// </summary>
internal int Baseline
{
get
{
return TextDpi.ToTextDpi(_line.Baseline);
}
}
/// <summary>
/// True if last line of paragraph
/// </summary>
internal bool EndOfParagraph
{
get
{
// If there are no Newline characters, it is not the end of paragraph.
if (_line.NewlineLength == 0)
{
return false;
}
// Since there are Newline characters in the line, do more expensive and
// accurate check.
return (((TextSpan<TextRun>)_runs[_runs.Count-1]).Value is ParagraphBreakRun);
}
}
/// <summary>
/// Length of the line including any synthetic characters.
/// This length is PTS frendly. PTS does not like 0 length lines.
/// </summary>
internal int SafeLength
{
get
{
return _line.Length;
}
}
/// <summary>
/// Length of the line excluding any synthetic characters.
/// </summary>
internal int ActualLength
{
get
{
return _line.Length - (EndOfParagraph ? _syntheticCharacterLength : 0);
}
}
/// <summary>
/// Length of the line excluding any synthetic characters and line breaks.
/// </summary>
internal int ContentLength
{
get
{
return _line.Length - _line.NewlineLength;
}
}
/// <summary>
/// Number of characters after the end of the line which may affect
/// line wrapping.
/// </summary>
internal int DependantLength
{
get
{
return _line.DependentLength;
}
}
/// <summary>
/// Was line truncated (forced broken)?
/// </summary>
internal bool IsTruncated
{
get
{
return _line.IsTruncated;
}
}
/// <summary>
/// Formatting result of the line.
/// </summary>
internal PTS.FSFLRES FormattingResult
{
get
{
PTS.FSFLRES formatResult = PTS.FSFLRES.fsflrOutOfSpace;
// If there are no Newline characters, we run out of space.
if (_line.NewlineLength == 0)
{
return formatResult;
}
// Since there are Newline characters in the line, do more expensive and
// accurate check.
TextRun run = ((TextSpan<TextRun>)_runs[_runs.Count - 1]).Value as TextRun;
if (run is ParagraphBreakRun)
{
formatResult = ((ParagraphBreakRun)run).BreakReason;
}
else if (run is LineBreakRun)
{
formatResult = ((LineBreakRun)run).BreakReason;
}
return formatResult;
}
}
#endregion Internal Properties
//-------------------------------------------------------------------
//
// Private Methods
//
//-------------------------------------------------------------------
#region Private Methods
/// <summary>
/// Returns true if there are any inline objects, false otherwise.
/// </summary>
private bool HasInlineObjects()
{
bool hasInlineObjects = false;
foreach (TextSpan<TextRun> textSpan in _runs)
{
if (textSpan.Value is InlineObjectRun)
{
hasInlineObjects = true;
break;
}
}
return hasInlineObjects;
}
/// <summary>
/// Returns bounds of an object/character at specified text index.
/// </summary>
/// <param name="cp">
/// Character index of an object/character
/// </param>
/// <param name="cch">
/// Number of positions occupied by object/character
/// </param>
/// <param name="flowDirection">
/// Flow direction of object/character
/// </param>
/// <returns></returns>
private Rect GetBoundsFromPosition(int cp, int cch, out FlowDirection flowDirection)
{
Rect rect;
// Calculate shift in line offset to render trailing spaces or avoid clipping text
double delta = TextDpi.FromTextDpi(CalculateUOffsetShift());
IList<TextBounds> textBounds;
if (_line.HasOverflowed && TextParagraph.Properties.TextTrimming != TextTrimming.None)
{
// We should not shift offset in this case
Invariant.Assert(DoubleUtil.AreClose(delta, 0));
System.Windows.Media.TextFormatting.TextLine line = _line.Collapse(GetCollapsingProps(_wrappingWidth, TextParagraph.Properties));
Invariant.Assert(line.HasCollapsed, "Line has not been collapsed");
textBounds = line.GetTextBounds(cp, cch);
}
else
{
textBounds = _line.GetTextBounds(cp, cch);
}
Invariant.Assert(textBounds != null && textBounds.Count == 1, "Expecting exactly one TextBounds for a single text position.");
IList<TextRunBounds> runBounds = textBounds[0].TextRunBounds;
if (runBounds != null)
{
Debug.Assert(runBounds.Count == 1, "Expecting exactly one TextRunBounds for a single text position.");
rect = runBounds[0].Rectangle;
}
else
{
rect = textBounds[0].Rectangle;
}
flowDirection = textBounds[0].FlowDirection;
rect.X = rect.X + delta;
return rect;
}
/// <summary>
/// Returns Line collapsing properties
/// </summary>
/// <param name="wrappingWidth">
/// Wrapping width for collapsed line.
/// </param>
/// <param name="paraProperties">
/// Paragraph properties
/// </param>
private TextCollapsingProperties GetCollapsingProps(double wrappingWidth, LineProperties paraProperties)
{
Invariant.Assert(paraProperties.TextTrimming != TextTrimming.None, "Text trimming must be enabled.");
TextCollapsingProperties collapsingProps;
if (paraProperties.TextTrimming == TextTrimming.CharacterEllipsis)
{
collapsingProps = new TextTrailingCharacterEllipsis(wrappingWidth, paraProperties.DefaultTextRunProperties);
}
else
{
collapsingProps = new TextTrailingWordEllipsis(wrappingWidth, paraProperties.DefaultTextRunProperties);
}
return collapsingProps;
}
/// <summary>
/// Perform depth-first search on a drawing tree to add all the glyph
/// runs to the collection
/// </summary>
/// <param name="drawing">
/// Drawing on which we perform DFS
/// </param>
/// <param name="glyphRunsCollection">
/// Glyph run collection.
/// </param>
/// <param name="cchGlyphRuns">
/// Character length of glyph run collection
/// </param>
private static void AddGlyphRunRecursive(Drawing drawing, List<GlyphRun> glyphRunsCollection, ref int cchGlyphRuns)
{
if (drawing is DrawingGroup group)
{
foreach (Drawing child in group.Children)
{
AddGlyphRunRecursive(child, glyphRunsCollection, ref cchGlyphRuns);
}
}
else
{
if (drawing is GlyphRunDrawing glyphRunDrawing)
{
// Add a glyph run
GlyphRun glyphRun = glyphRunDrawing.GlyphRun;
if (glyphRun != null)
{
cchGlyphRuns += (glyphRun.Characters == null ? 0 : glyphRun.Characters.Count);
glyphRunsCollection.Add(glyphRun);
}
}
}
}
/// <summary>
/// Returns amount of shift for X-offset to render trailing spaces
/// </summary>
internal int CalculateUOffsetShift()
{
int width;
int trailingSpacesDelta = 0;
// Calculate amount by which to to move line back if trailing spaces are rendered
if (IsUOffsetAdjusted)
{
width = TextDpi.ToTextDpi(_line.WidthIncludingTrailingWhitespace);
trailingSpacesDelta = TextDpi.ToTextDpi(_line.Width) - width;
Invariant.Assert(trailingSpacesDelta <= 0);
}
else
{
width = TextDpi.ToTextDpi(_line.Width);
trailingSpacesDelta = 0;
}
// Calculate amount to shift line forward in case we are clipping the front of the line.
// If line is showing ellipsis do not perform this check since we should not be clipping the front
// of the line anyway
int widthDelta = 0;
if ((_textAlignment == TextAlignment.Center || _textAlignment == TextAlignment.Right) && !ShowEllipses)
{
if (width > TextDpi.ToTextDpi(_wrappingWidth))
{
widthDelta = width - TextDpi.ToTextDpi(_wrappingWidth);
}
else
{
widthDelta = 0;
}
}
int totalShift;
if (_textAlignment == TextAlignment.Center)
{
// Divide shift by two to center line
totalShift = (int)((widthDelta + trailingSpacesDelta) / 2);
}
else
{
totalShift = widthDelta + trailingSpacesDelta;
}
return totalShift;
}
#endregion Private methods
//-------------------------------------------------------------------
//
// Private Properties
//
//-------------------------------------------------------------------
#region Private Properties
/// <summary>
/// True if line ends in hard break
/// </summary>
private bool HasLineBreak
{
get
{
return (_line.NewlineLength > 0);
}
}
/// <summary>
/// True if line's X-offset needs adjustment to render trailing spaces
/// </summary>
private bool IsUOffsetAdjusted
{
get
{
return ((_textAlignment == TextAlignment.Right || _textAlignment == TextAlignment.Center) && IsWidthAdjusted);
}
}
/// <summary>
/// True if line's width is adjusted to include trailing spaces. For right and center alignment we need to
/// adjust line offset as well, but for left alignment we need to only make a width asjustment
/// </summary>
private bool IsWidthAdjusted
{
get
{
bool adjusted = false;
// Trailing spaces rendered only around hard breaks
if (HasLineBreak || EndOfParagraph)
{
// Lines with ellipsis are not shifted because ellipsis would not appear after trailing spaces
if (!ShowEllipses)
{
adjusted = true;
}
}
return adjusted;
}
}
/// <summary>
/// True if eliipsis is displayed in the line
/// </summary>
private bool ShowEllipses
{
get
{
if (TextParagraph.Properties.TextTrimming == TextTrimming.None)
{
return false;
}
if (_line.HasOverflowed)
{
return true;
}
return false;
}
}
/// <summary>
/// Text Paragraph this line is formatted for
/// </summary>
private TextParagraph TextParagraph
{
get
{
return _paraClient.Paragraph as TextParagraph;
}
}
#endregion Private Properties
//-------------------------------------------------------------------
//
// Private Fields
//
//-------------------------------------------------------------------
#region Private Fields
/// <summary>
/// TextFormatter host
/// </summary>
private readonly TextFormatterHost _host;
/// <summary>
/// Character position at the beginning of text paragraph. All DCPs
/// of the line are relative to this value.
/// </summary>
private readonly int _cpPara;
/// <summary>
/// Line formatting context. Valid only during formatting.
/// </summary>
private FormattingContext _formattingContext;
/// <summary>
/// Text line objects
/// </summary>
private System.Windows.Media.TextFormatting.TextLine _line;
/// <summary>
/// Cached run list. This list needs to be in sync with _line object.
/// Every time the line is recreated, this list needs to be updated.
/// </summary>
private IList<TextSpan<TextRun>> _runs;
/// <summary>
/// Character position at the beginning of the line.
/// </summary>
private int _dcp;
/// <summary>
/// Line wrapping width
/// </summary>
private double _wrappingWidth;
/// <summary>
/// Track width (line width ignoring floats)
/// </summary>
private double _trackWidth = Double.NaN;
/// <summary>
/// Is text mirrored?
/// </summary>
private bool _mirror;
/// <summary>
/// Text indent. 0 for all lines except the first line, which maybe have non-zero indent.
/// </summary>
private double _indent;
/// <summary>
/// TextAlignment of owner
/// </summary>
private TextAlignment _textAlignment;
#endregion Private Fields
// ------------------------------------------------------------------
//
// FormattingContext Class
//
// ------------------------------------------------------------------
#region FormattingContext Class
/// <summary>
/// Text line formatting context
/// </summary>
internal class FormattingContext
{
internal FormattingContext(bool measureMode, bool clearOnLeft, bool clearOnRight, TextRunCache textRunCache)
{
MeasureMode = measureMode;
ClearOnLeft = clearOnLeft;
ClearOnRight = clearOnRight;
TextRunCache = textRunCache;
LineFormatLengthTarget = -1;
}
internal TextRunCache TextRunCache;
internal bool MeasureMode;
internal bool ClearOnLeft;
internal bool ClearOnRight;
internal int LineFormatLengthTarget;
}
#endregion FormattingContext Class
}
}
|