Code:
/ Net / Net / 3.5.50727.3053 / DEVDIV / depot / DevDiv / releases / Orcas / SP / wpf / src / Core / CSharp / System / Windows / Media / GlyphRun.cs / 1 / GlyphRun.cs
//---------------------------------------------------------------------------- // // Copyright (c) Microsoft Corporation. All rights reserved. // // Description: The GlyphRun object represents a sequence of glyphs from a single face of a single font at a single size, // and with a single rendering style. // // See specs at // http://team/sites/Avalon/Specs/Glyphs%20element%20and%20GlyphRun%20object.htm // http://team/sites/Avalon/Specs/Glyph%20Run%20hit%20testing%20and%20caret%20placement%20API.htm // // // History: // 02/18/2003 : mleonov - Created // //--------------------------------------------------------------------------- // Enable presharp pragma warning suppress directives. #pragma warning disable 1634, 1691 using System; using System.Collections; using System.Collections.Generic; using System.ComponentModel; using System.Diagnostics; using System.Globalization; using System.Text; using System.Windows; using System.Windows.Media; using System.Windows.Media.Converters; using System.Windows.Media.Composition; using System.Windows.Media.TextFormatting; using System.Windows.Markup; using System.Runtime.InteropServices; using MS.Internal; using MS.Internal.FontCache; using MS.Internal.FontFace; using MS.Internal.TextFormatting; using MS.Utility; using System.Security; using System.Security.Permissions; using SR=MS.Internal.PresentationCore.SR; using SRID=MS.Internal.PresentationCore.SRID; namespace System.Windows.Media { ////// The GlyphRun object represents a sequence of glyphs from a single face of a single font at a single size, /// and with a single rendering style. /// // public class GlyphRun : DUCE.IResource, ISupportInitialize { //----------------------------------------------------- // // Constructors // //----------------------------------------------------- #region Constructors ////// Construct an uninitialized GlyphRun object. Caller should call ISupportInitialize.BeginInit() /// to begin initialization and call ISupportInitialize.EndInit() to finish the initialization. /// The GlyphRun does not support all the operations until it is fully initialized. /// public GlyphRun() { } ////// Constructs a new GlyphRun object. /// /// GlyphTypeface of the GlyphRun object /// Bidi level of the GlyphRun object /// Set to true to display the GlyphRun sideways /// Font rendering size in drawing surface units (96ths of an inch). /// The list of font indices that represent glyphs in this run. /// Origin of the first glyph in the run. /// The glyph is placed so that the leading edge of its advance vector /// and its baseline intersect this point. /// /// The list of advance widths, one for each glyph in GlyphIndices. /// The nominal origin of the nth glyph (n > 0) in the run is the nominal origin /// of the n-1th glyph plus the n-1th advance width added along the runs advance vector. /// Base glyphs generally have a non-zero advance width, combining glyphs generally have a zero advance width. /// /// The list of glyph offsets. Added to the nominal glyph origin calculated above to generate the final origin for the glyph. /// Base glyphs generally have a glyph offset of (0,0), combining glyphs generally have an offset /// that places them correctly on top of the nearest preceeding base glyph. /// /// Characters represented by this glyphrun /// /// Identifies a specific device font for which the GlyphRun has been optimized. When a GlyphRun is /// being rendered on a device that has built-in support for this named font, then the GlyphRun should be rendered using a /// possibly device specific mechanism for selecting that font, and by sending the Unicode codepoints rather than the /// glyph indices. When rendering onto a device that does not include built-in support for the named font, /// this property should be ignored. /// /// The list that maps characters in the glyph run to glyph indices. /// There is one entry per character in Characters list. /// Each value gives the offset of the first glyph in GlyphIndices /// that represents the corresponding character in Characters. /// Where multiple characters map to a single glyph, or to a glyph group /// that cannot be broken down to map exactly to individual characters, /// the entries for all the characters have the same value: /// the offset of the first glyph that represents this group of characters. /// If the list is null or empty, sequential 1 to 1 mapping is assumed. /// /// A list of caret stops for the glyphs /// Language of the GlyphRun [CLSCompliant(false)] public GlyphRun( GlyphTypeface glyphTypeface, int bidiLevel, bool isSideways, double renderingEmSize, IListglyphIndices, Point baselineOrigin, IList advanceWidths, IList glyphOffsets, IList characters, string deviceFontName, IList clusterMap, IList caretStops, XmlLanguage language ) { // Suppress PRESharp warning that glyphIndices and advanceWidths are not validated and can be null. // They can indeed be null, but that's perfectly OK. An explicit null check in the constructor is // not required. #pragma warning suppress 56506 Initialize( glyphTypeface, bidiLevel, isSideways, renderingEmSize, glyphIndices, baselineOrigin, advanceWidths, glyphOffsets, characters, deviceFontName, clusterMap, caretStops, language, true // throwIfOverflow ); // GlyphRunFlags.CacheInkBounds enanbles ink bounding box caching. Bounding box caching would cost // 32 bytes per GlyphRun. We do not want to enable it for all cases possible working set increase. // For Line layout, ink bounding box is only used a few times, so caching is disabled because it will // go through TryCreate below. Memory cost: 1 pointer. // For loading XPS in which bounding box calculation are called a lot in hit testing, Glyphs.cs will // call this constructor, which enables caching. Memory cost: 1 pointer + boxed Rect. // If we late decide it's worthwhile to cache for all, memory cost can be reduced to one Rect (32-bytes). // If we decide single precision is good enough, it can be reduced to 16 bytes. _flags |= GlyphRunFlags.CacheInkBounds; } /// /// Creates a new GlyphRun object. This method is similar to the constructor with /// the same argument list except that it returns null instead of throwing an /// exception if the GlyphRun area or a coordinate exceed the maximum value. /// internal static GlyphRun TryCreate( GlyphTypeface glyphTypeface, int bidiLevel, bool isSideways, double renderingEmSize, IListglyphIndices, Point baselineOrigin, IList advanceWidths, IList glyphOffsets, IList characters, string deviceFontName, IList clusterMap, IList caretStops, XmlLanguage language ) { GlyphRun glyphRun = new GlyphRun(); // Suppress PRESharp warning that glyphIndices and advanceWidths are not validated and can be null. // They can indeed be null, but that's perfectly OK. An explicit null check in the constructor is // not required. #pragma warning suppress 56506 glyphRun.Initialize( glyphTypeface, bidiLevel, isSideways, renderingEmSize, glyphIndices, baselineOrigin, advanceWidths, glyphOffsets, characters, deviceFontName, clusterMap, caretStops, language, false // throwIfOverflow ); // Cached GlyphRun bounds are needed to pass to the render thread glyphRun._flags |= GlyphRunFlags.CacheInkBounds; if (glyphRun.IsInitialized) return glyphRun; else return null; } private void Initialize( GlyphTypeface glyphTypeface, int bidiLevel, bool isSideways, double renderingEmSize, IList glyphIndices, Point baselineOrigin, IList advanceWidths, IList glyphOffsets, IList characters, string deviceFontName, IList clusterMap, IList caretStops, XmlLanguage language, bool throwOnOverflow ) { // The default branch prediction rules for modern processors specify that forward branches // are not to be taken. If the branch is in fact taken, all of the speculatively executed code // must be discarded, the processor pipeline flushed, and then reloaded. This results in a // processor stall of at least 42 cycles for the P4 Northwood for each mis-predicted branch. // The deeper the processor pipeline the higher the cost, i.e. Prescott processors. // Checking for multiple incorrect parameters in a method with high call count like this one can // easily add significant overhead for no reason. Note that the C# compiler should be able to make // reasonable assumptions about branches that throw exceptions, but the current whidbey // implemenation is weak in this regard. Also the current IBC tools are unable to add branch // prediction hints to improve behavior based on run time information. Also note that adding // branch prediction hints increases code size by a byte per branch and doing this in every // method that is coded without default branch prediction behavior in mind would add an // unacceptable amount of working set. if ((glyphTypeface != null) && (glyphIndices != null) && (advanceWidths != null) && (renderingEmSize >= 0.0) && (glyphIndices.Count > 0) && (glyphIndices.Count <= MaxGlyphCount) && (advanceWidths.Count == glyphIndices.Count) && ((glyphOffsets == null) || ((glyphOffsets != null) && (glyphOffsets.Count != 0) && (glyphOffsets.Count == glyphIndices.Count)))) { // Set member variables here, // so that GlyphRun properties can be calculated in advanced validation code. _glyphIndices = glyphIndices; _characters = characters; _clusterMap = clusterMap; _baselineOrigin = baselineOrigin; _renderingEmSize = renderingEmSize; _advanceWidths = advanceWidths; _glyphOffsets = glyphOffsets; _glyphTypeface = glyphTypeface; _flags = (isSideways ? GlyphRunFlags.IsSideways : GlyphRunFlags.None); _bidiLevel = bidiLevel; _caretStops = caretStops; _language = language; _deviceFontName = deviceFontName; if (characters != null && characters.Count != 0) { if (clusterMap != null && clusterMap.Count != 0) { if (clusterMap.Count == characters.Count) { // Perform some simple cluster map validation. // First entry should be zero, the entries should be monotonic and shouldn't point outside of the glyph indices range. if (clusterMap[0] == 0) { int glyphCount = GlyphCount; int mapCount = clusterMap.Count; ushort previous = clusterMap[0]; for (int i = 1; i < mapCount; ++i) { ushort current = clusterMap[i]; if ((current >= previous) && (current < glyphCount)) { previous = current; } else { if (clusterMap[i] < clusterMap[i - 1]) throw new ArgumentException(SR.Get(SRID.ClusterMapEntriesShouldNotDecrease), "clusterMap"); if (clusterMap[i] >= GlyphCount) throw new ArgumentException(SR.Get(SRID.ClusterMapEntryShouldPointWithinGlyphIndices), "clusterMap"); } } } else { throw new ArgumentException(SR.Get(SRID.ClusterMapFirstEntryMustBeZero), "clusterMap"); } } else { throw new ArgumentException(SR.Get(SRID.CollectionNumberOfElementsShouldBeEqualTo, characters.Count), "clusterMap"); } } else { if (GlyphCount != characters.Count) throw new ArgumentException(SR.Get(SRID.CollectionNumberOfElementsShouldBeEqualTo, GlyphCount), "clusterMap"); } } if (caretStops != null && caretStops.Count != 0) { if (caretStops.Count != CodepointCount + 1) throw new ArgumentException(SR.Get(SRID.CollectionNumberOfElementsShouldBeEqualTo, CodepointCount + 1), "caretStops"); } if (isSideways && (bidiLevel & 1) != 0) throw new ArgumentException(SR.Get(SRID.SidewaysRTLTextIsNotSupported)); { // Check given arguments against big numbers. // The purpose of this block is to ensure that we'll // never get arithmetic overflow on rendering, and reduce // the risk of memory overflow. // Too big numbers will be discovered right here and cause // throwing the exception. // Coordinates are submitted to rendering in normalized form, // i.e. divided by renderingEmSize. // On rendering, coordinate values are multiplied by rasterization scale // that is known not to exceed sc_uGeometryThreshold * sc_uMaxOvescale. // Obtained values may be shifted by Origin and Advance values taken from // GlyphBitmap. The latter two values are shifted 16-bit short. The sum is then // converted to integer value using CFloatFPU::SmallRound routine // that accepts numbers in range -OverscaledCoordMax <= N <= OverscaledCoordMax. // Hence we have available maximum for given coordinate values: double coordMax = renderingEmSize * ((double)(OverscaledCoordMax - 0x10000)) / (GeometryThreshold * MaxOvescale); double glyphXMin = 0.0; double glyphXMax = 0.0; double glyphYMin = 0.0; double glyphYMax = 0.0; // First, calculate the smallest rectangle containing anchor points // of all of the glyphs. if (glyphOffsets != null) { double glyphX = glyphOffsets[0].X; double glyphY = glyphOffsets[0].Y; glyphXMin = glyphX; glyphXMax = glyphX; glyphYMin = glyphY; glyphYMax = glyphY; if ((Math.Abs(glyphX) <= coordMax) && (Math.Abs(glyphY) <= coordMax)) { double positionX = 0; int glyphCount = GlyphCount; for (int i = 1; i < glyphCount; i++) { positionX += advanceWidths[i - 1]; if (Math.Abs(positionX) <= coordMax) { glyphX = positionX; glyphY = 0; double glyphOffsetX = glyphOffsets[i].X; double glyphOffsetY = glyphOffsets[i].Y; if (Math.Abs(glyphOffsetX) <= coordMax && (Math.Abs(glyphOffsetY) <= coordMax)) { glyphX += glyphOffsetX; glyphY += glyphOffsetY; if (glyphX < glyphXMin) glyphXMin = glyphX; else if (glyphX > glyphXMax) glyphXMax = glyphX; if (glyphY < glyphYMin) glyphYMin = glyphY; else if (glyphY > glyphYMax) glyphYMax = glyphY; // Continue looping, no error continue; } } // We will get here if either range check above fails if (throwOnOverflow) ReportCoordinateOverflow(i, renderingEmSize, coordMax); else return; } } else { if (throwOnOverflow) ReportCoordinateOverflow(0, renderingEmSize, coordMax); else return; } } else { double glyphX = 0.0; int glyphCount = GlyphCount; for (int i = 1; i < glyphCount; i++) { glyphX += advanceWidths[i - 1]; if (Math.Abs(glyphX) <= coordMax) { if (glyphX < glyphXMin) glyphXMin = glyphX; else if (glyphX > glyphXMax) glyphXMax = glyphX; } else if (throwOnOverflow) { ReportCoordinateOverflow(i, renderingEmSize, coordMax); } else { return; } } } // We have the rectangle bounds (glyphXMin, glyphXMax, glyphYMin, glyphYMax) // in local coordinate space. Area occupied by glyph run is bigger because // glyphs have sizes. Note that glyph anchor point might be out of glyph // bounding rectangle, due to overhang. We allow 2*renderingEmSize offset from // glyph anchor point to any of four lines that compose glyph bounding rectangle. // // Now we can estimate the area occupied by glyph run. double relativeWidth = (glyphXMax - glyphXMin) / renderingEmSize + 4; double relativeHeight = (glyphYMax - glyphYMin) / renderingEmSize + 4; double relativeArea = relativeWidth * relativeHeight; double scaleMax = GeometryThreshold * MaxOvescale; double relativeAreaMax = MaxOverscaledBitsInGlyphRun / (scaleMax * scaleMax); // Reject the glyph run that occupies too big area. if (relativeArea > relativeAreaMax) { if (throwOnOverflow) ReportAreaOverflow(relativeArea, relativeAreaMax); else return; } } } else { if (DoubleUtil.IsNaN(renderingEmSize)) throw new ArgumentOutOfRangeException("renderingEmSize", SR.Get(SRID.ParameterValueCannotBeNaN)); if (renderingEmSize < 0.0) throw new ArgumentOutOfRangeException("renderingEmSize", SR.Get(SRID.ParameterValueCannotBeNegative)); if (glyphTypeface == null) throw new ArgumentNullException("glyphTypeface"); if (glyphIndices == null) throw new ArgumentNullException("glyphIndices"); if (glyphIndices.Count <= 0) throw new ArgumentException(SR.Get(SRID.CollectionNumberOfElementsMustBeGreaterThanZero), "glyphIndices"); if (glyphIndices.Count > MaxGlyphCount) { throw new ArgumentException(SR.Get(SRID.CollectionNumberOfElementsMustBeLessOrEqualTo, MaxGlyphCount), "glyphIndices"); } if (advanceWidths == null) throw new ArgumentNullException("advanceWidths"); if (advanceWidths.Count != glyphIndices.Count) throw new ArgumentException(SR.Get(SRID.CollectionNumberOfElementsShouldBeEqualTo, glyphIndices.Count), "advanceWidths"); if (glyphOffsets != null && glyphOffsets.Count != 0 && glyphOffsets.Count != glyphIndices.Count) throw new ArgumentException(SR.Get(SRID.CollectionNumberOfElementsShouldBeEqualTo, glyphIndices.Count), "glyphOffsets"); // We should've caught all invalid cases above and thrown appropriate exceptions. Invariant.Assert(false); } IsInitialized = true; // The glyphrun is completely initialized } private void ReportCoordinateOverflow(int index, double renderingEmSize, double coordMax) { string message = SR.Get(SRID.GlyphCoordinateTooBig); message = string.Format(CultureInfo.CurrentCulture, message, index, renderingEmSize, coordMax); throw new OverflowException(message); } private void ReportAreaOverflow(double relativeArea, double relativeAreaMax) { string message = SR.Get(SRID.GlyphAreaTooBig); message = string.Format(CultureInfo.CurrentCulture, message, relativeArea, relativeAreaMax); throw new OverflowException(message); } #endregion Constructors //------------------------------------------------------ // // Public Methods // //----------------------------------------------------- #region Public Methods /// /// Given a character hit, computes the offset from the leading edge of the glyph run /// to the leading or trailing edge of a caret stop containing the character hit. /// If the glyph run is not hit testable, the distance of 0.0 is returned. /// /// Character hit to compute the distance to. ///The offset from the leading edge of the glyph run /// to the leading or trailing edge of a caret stop containing the character hit. ////// The input character hit is outside of the range specified by the glyph run Unicode string. /// public double GetDistanceFromCaretCharacterHit(CharacterHit characterHit) { CheckInitialized(); // This can only be called on fully initialized GlyphRun IListcaretStops = CaretStops != null && CaretStops.Count != 0 ? CaretStops : new DefaultCaretStopList(CodepointCount); if (characterHit.FirstCharacterIndex < 0 || characterHit.FirstCharacterIndex > CodepointCount) throw new ArgumentOutOfRangeException("characterHit"); int caretStopIndex, codePointsUntilNextStop; FindNearestCaretStop( characterHit.FirstCharacterIndex, caretStops, out caretStopIndex, out codePointsUntilNextStop); // Not a hit testable glyph run. if (caretStopIndex == -1) return 0.0; // Trailing edge of a caret stop that doesn't have a corresponding valid next caret stop. if (codePointsUntilNextStop == -1 && characterHit.TrailingLength != 0) { return 0.0; } // Code point we are measuring the distance to. int caretCodePoint = characterHit.TrailingLength == 0 ? caretStopIndex : caretStopIndex + codePointsUntilNextStop; double distance = 0.0; // Sum up glyph advance widths until the caret code point. IList clusterMap = ClusterMap; if (clusterMap == null) clusterMap = new DefaultClusterMap(CodepointCount); int clusterCodepointStart = 0; int currentCodepoint = clusterCodepointStart; IList advances = AdvanceWidths; for (;;) { ++currentCodepoint; if (currentCodepoint >= clusterMap.Count || clusterMap[currentCodepoint] != clusterMap[clusterCodepointStart]) { // We reached the beginning of the next cluster or the end of the glyph run. // If the codepoint is within the cluster, calculate the partial width and abort the loop. // If the codepoint is past the cluster, accumulate the whole cluster advance width and move on. double clusterWidth = 0; int clusterGlyphEnd; if (currentCodepoint >= clusterMap.Count) clusterGlyphEnd = advances.Count; else clusterGlyphEnd = clusterMap[currentCodepoint]; for (int i = clusterMap[clusterCodepointStart]; i < clusterGlyphEnd; ++i) clusterWidth += advances[i]; if (caretCodePoint < currentCodepoint || currentCodepoint >= clusterMap.Count) { // The caret code point is within a cluster or we are past the end of the run, // sum all glyph advance widths in the cluster // and multiply the result by (caretCodePoint / number of codepoints in the cluster). clusterWidth *= (double)(caretCodePoint - clusterCodepointStart) / (currentCodepoint - clusterCodepointStart); distance += clusterWidth; break; } // The codepoint is past the cluster, accumulate the whole cluster advance width and move on. distance += clusterWidth; clusterCodepointStart = currentCodepoint; } } return distance; } /// /// Given an offset from the leading edge of the glyph run, computes the caret character hit /// that contains the offset. The out bool IsInside parameter describes whether the character hit /// is inside the glyph run. If the hit is outside the glyph run, the character hit represents /// the closest caret character hit within the glyph run. /// /// Distance to compute character hit for. /// isInside is set to true when the character hit /// is inside the glyph run, and to false otherwise. ///The character hit that is closest to the input distance. public CharacterHit GetCaretCharacterHitFromDistance(double distance, out bool isInside) { CheckInitialized(); // This can only be called on fully initialized GlyphRun // Navigate the caret stop array and find a pair of caret stops that contains the distance. IListadvances = AdvanceWidths; IList caretStops = CaretStops != null && CaretStops.Count != 0 ? CaretStops : new DefaultCaretStopList(CodepointCount); IList clusterMap = ClusterMap; if (clusterMap == null) clusterMap = new DefaultClusterMap(CodepointCount); // The following two variables describe the closest caret stop to the left of the input distance. int firstStopIndex = -1; double firstStopAdvance = 0.0; // The following variable describes the closest caret stop to the right of the input distance. int secondStopIndex = -1; // Accumulated advance width just before the current cluster. double currentAdvance = 0.0; // Start index of the cluster we're in. int currentClusterStart = 0; // Since the caretStops array contains clusterMap.Count + 1 elements, // we need to be careful before dereferencing i in the loop body. for (int i = 1; i < caretStops.Count; ++i) { if (i < clusterMap.Count && clusterMap[i] == clusterMap[currentClusterStart]) continue; // We reached the end of an (n:m) cluster. // First, accumulate the overall cluster advance width by summing m glyph advances. ushort lastGlyphInCluster = i < clusterMap.Count ? clusterMap[i] : (ushort)advances.Count; Debug.Assert(clusterMap[currentClusterStart] < lastGlyphInCluster); double clusterAdvance = 0.0; for (int j = clusterMap[currentClusterStart]; j < lastGlyphInCluster; ++j) clusterAdvance += advances[j]; // The overall advance is divided evenly by n code points. clusterAdvance /= i - currentClusterStart; // Go through the individual caret stops and compare them against the input distance for (int j = currentClusterStart; j < i; ++j) { if (caretStops[j]) { if (currentAdvance <= distance) { firstStopIndex = j; firstStopAdvance = currentAdvance; } else { // We found a caret stop to the right of the input distance, // so we're done with enumerating. secondStopIndex = j; goto SecondStopFound; } } currentAdvance += clusterAdvance; } currentClusterStart = i; } // The last iteration is interesting. Because inside the above loop we only look at the caret stops up until i-1, // and there may or may not be a caret stop at the end of a glyph run, // we need to check the last caret stop value and the distance. // The code before SecondStopFound is essentially the reduced version of the loop body above when i == caretStops.Count. // We could modify the loop, but this would result in additional special cases. if (caretStops[caretStops.Count - 1]) { if (currentAdvance > distance) secondStopIndex = caretStops.Count - 1; } SecondStopFound: // First stop is described by firstStopIndex, firstStopAdvance. // Second stop is described by secondStopIndex, currentAdvance. // If both indices are equal to -1, then all caret stop entries except the very last one are set to false. // If the last one is also set to false, the glyph run is not hit testable at all. // If the last one is set to true, we return CharacterHit corresponding to that last caret stop. if (firstStopIndex == -1 && secondStopIndex == -1) { isInside = false; if (caretStops[caretStops.Count - 1]) return new CharacterHit(caretStops.Count - 1, 0); else return new CharacterHit(0, 0); } // Check for case when the first stop is not valid. // This happens when the hit is to the left of the first caret stop. if (firstStopIndex == -1) { isInside = false; // Leading edge of the second stop. return new CharacterHit(secondStopIndex, 0); } // Check for case when the second stop is not valid. // This happens when the hit is to the right of the last caret stop. if (secondStopIndex == -1) { isInside = false; // Trailing edge of the first stop. return new CharacterHit(firstStopIndex, caretStops.Count - 1 - firstStopIndex); } isInside = true; if (distance <= (firstStopAdvance + currentAdvance) / 2.0) { // Leading edge of the first stop. return new CharacterHit(firstStopIndex, 0); } else { // Trailing edge of the first stop. return new CharacterHit(firstStopIndex, secondStopIndex - firstStopIndex); } } /// /// Computes the next valid caret character hit in the logical direction. /// If no further navigation is possible, the returned hit result is the same as input value. /// /// The character hit to compute next hit value for. ///The next valid caret character hit in the logical direction, or /// the input value if no further navigation is possible. public CharacterHit GetNextCaretCharacterHit(CharacterHit characterHit) { CheckInitialized(); // This can only be called on fully initialized GlyphRun IListcaretStops = CaretStops != null && CaretStops.Count != 0 ? CaretStops : new DefaultCaretStopList(CodepointCount); if (characterHit.FirstCharacterIndex < 0 || characterHit.FirstCharacterIndex > CodepointCount) throw new ArgumentOutOfRangeException("characterHit"); int caretStopIndex, codePointsUntilNextStop; FindNearestCaretStop( characterHit.FirstCharacterIndex, caretStops, out caretStopIndex, out codePointsUntilNextStop); // Not a hit testable run, or no next caret code point. if (caretStopIndex == -1 || codePointsUntilNextStop == -1) return characterHit; // If we are at the leading edge, move to the trailing edge of the same code point. if (characterHit.TrailingLength == 0) return new CharacterHit(caretStopIndex, codePointsUntilNextStop); // If the next caret stop is within the glyph run, // move to the trailing edge of it. int nextCaretStopIndex, nextCodePointsUntilNextStop; FindNearestCaretStop( caretStopIndex + codePointsUntilNextStop, caretStops, out nextCaretStopIndex, out nextCodePointsUntilNextStop); // See if the next caret stop is within the glyph run. // If not, no navigation is possible. if (nextCodePointsUntilNextStop == -1) return characterHit; return new CharacterHit(nextCaretStopIndex, nextCodePointsUntilNextStop); } /// /// Computes the previous valid caret character hit in the logical direction. /// If no further navigation is possible, the returned hit result is the same as input value. /// /// The character hit to compute previous hit value for. ///The previous valid caret character hit in the logical direction, or /// the input value if no further navigation is possible. public CharacterHit GetPreviousCaretCharacterHit(CharacterHit characterHit) { CheckInitialized(); // This can only be called on fully initialized GlyphRun IListcaretStops = CaretStops != null && CaretStops.Count != 0 ? CaretStops : new DefaultCaretStopList(CodepointCount); if (characterHit.FirstCharacterIndex < 0 || characterHit.FirstCharacterIndex > CodepointCount) throw new ArgumentOutOfRangeException("characterHit"); int caretStopIndex, codePointsUntilNextStop; FindNearestCaretStop( characterHit.FirstCharacterIndex, caretStops, out caretStopIndex, out codePointsUntilNextStop); if (caretStopIndex == -1) return characterHit; // If we are at the trailing edge, move to the leading edge of the same code point. if (characterHit.TrailingLength != 0) return new CharacterHit(caretStopIndex, 0); // Find the previous caret stop. int previousCaretStopIndex; FindNearestCaretStop( caretStopIndex - 1, caretStops, out previousCaretStopIndex, out codePointsUntilNextStop); // No previous hit, return the original one. if (previousCaretStopIndex == -1 || previousCaretStopIndex == caretStopIndex) return characterHit; return new CharacterHit(previousCaretStopIndex, 0); } #endregion Public Methods //------------------------------------------------------ // // Public Properties // //------------------------------------------------------ #region Public Properties /// /// Advance width from origin of first glyph to far alignment edge of last glyph. /// private double AdvanceWidth { get { double advance = 0; if (_advanceWidths != null) { foreach(double glyphAdvance in _advanceWidths) advance += glyphAdvance; } return advance; } } ////// Distance from the GlyphRun origin to the top of the alignment box. /// private double Ascent { get { // for sideways text, origin is in the middle of the character cell if (IsSideways) return _renderingEmSize * _glyphTypeface.Height / 2.0; return _glyphTypeface.Baseline * _renderingEmSize; } } ////// Distance from top to bottom of alignment box. /// private double Height { get { return _glyphTypeface.Height * _renderingEmSize; } } ////// The baseline origin of the glyph run /// public Point BaselineOrigin { get { CheckInitialized(); return _baselineOrigin; } set { CheckInitializing(); // This can only be set during initialization. _baselineOrigin = value; } } ////// Em size used for rendering. /// public double FontRenderingEmSize { get { CheckInitialized(); return _renderingEmSize; } set { CheckInitializing(); // This can only be set during initialization. _renderingEmSize = value; } } ////// Returns GlyphTypeface for this object. /// public GlyphTypeface GlyphTypeface { get { CheckInitialized(); return _glyphTypeface; } set { CheckInitializing(); // This can only be set during initialization. if (value == null) { throw new ArgumentNullException("value"); } _glyphTypeface = value; } } ////// Determines LTR/RTL reading order and bidi nesting. /// ///The value of bidirectional nesting level. public int BidiLevel { get { CheckInitialized(); return _bidiLevel; } set { CheckInitializing(); // This can only be set during initialization. _bidiLevel = value; } } ////// Returns whether the glyph run is left to right or right to left. /// ///true for LTR, false for RTL. private bool IsLeftToRight { get { return (_bidiLevel & 1) == 0; } } ////// Specifies whether to rotate characters/glyphs 90 degrees anti-clockwise /// and use vertical baseline positioning metrics. /// ///true if the rotation should be applied, false otherwise. public bool IsSideways { get { CheckInitialized(); return (_flags & GlyphRunFlags.IsSideways) != 0; } set { CheckInitializing(); // This can only be set during initialization. if (value) { _flags |= GlyphRunFlags.IsSideways; } else { _flags &= (~GlyphRunFlags.IsSideways); } } } ////// Returns caret stops list for this GlyphRun or null if there is a caret stop for every UTF16 codepoint. /// [CLSCompliant(false)] [TypeConverter(typeof(BoolIListConverter))] public IListCaretStops { get { CheckInitialized(); return _caretStops; } set { CheckInitializing(); // This can only be set during initialization. // The list can be null, empty or non-empty list. // The consistency with other lists would be checked at EndInit() time. _caretStops = value; } } /// /// Returns whether there are any valid caret character hits within the glyph run. /// public bool IsHitTestable { get { CheckInitialized(); // This can only be called on fully initialized GlyphRun if (CaretStops == null || CaretStops.Count == 0) { // When CaretStops property is omitted, there is a caret stop for every UTF16 code point. return true; } foreach (bool caretStop in CaretStops) { if (caretStop) return true; } return false; } } ////// The list that maps characters in the glyph run to glyph indices. /// There is one entry per character in Characters list. /// Each value gives the offset of the first glyph in GlyphIndices /// that represents the corresponding character in Characters. /// Where multiple characters map to a single glyph, or to a glyph group /// that cannot be broken down to map exactly to individual characters, /// the entries for all the characters have the same value: /// the offset of the first glyph that represents this group of characters. /// If the list is null or empty, sequential 1 to 1 mapping is assumed. /// [CLSCompliant(false)] [TypeConverter(typeof(UShortIListConverter))] public IListClusterMap { get { CheckInitialized(); return _clusterMap; } set { CheckInitializing(); // This can only be set during initialization. // The list can be null, empty or non-empty list. // The consistency with other lists would be checked at EndInit() time. _clusterMap = value; } } /// /// Returns the list of UTF16 code points that represent the Unicode content of the glyph run. /// [CLSCompliant(false)] [TypeConverter(typeof(CharIListConverter))] public IListCharacters { get { CheckInitialized(); return _characters; } set { CheckInitializing(); // This can only be set during initialization. // The list can be null, empty or non-empty list. // The consistency with other lists would be checked at EndInit() time. _characters = value; } } /// /// Array of 16 bit glyph numbers that represent this run. /// [CLSCompliant(false)] [TypeConverter(typeof(UShortIListConverter))] public IListGlyphIndices { get { CheckInitialized(); return _glyphIndices; } set { CheckInitializing(); // This can only be set during initialization. // The list must be non-empty list. // The consistency with other lists would be checked at EndInit() time. if (value == null) throw new ArgumentNullException("value"); if (value.Count <= 0) throw new ArgumentException(SR.Get(SRID.CollectionNumberOfElementsMustBeGreaterThanZero), "GlyphIndices"); _glyphIndices = value; } } /// /// The list of advance widths, one for each glyph in GlyphIndices. /// The nominal origin of the nth glyph in the run (n>0) is the nominal origin /// of the n-1th glyph plus the n-1th advance width added along the runs advance vector. /// Base glyphs generally have a non-zero advance width, combining glyphs generally have a zero advance width. /// [CLSCompliant(false)] [TypeConverter(typeof(DoubleIListConverter))] public IListAdvanceWidths { get { CheckInitialized(); return _advanceWidths; } set { CheckInitializing(); // This can only be set during initialization. // The list must be non-empty list. // The consistency with other lists would be checked at EndInit() time. if (value == null) throw new ArgumentNullException("value"); if (value.Count <= 0) throw new ArgumentException(SR.Get(SRID.CollectionNumberOfElementsMustBeGreaterThanZero), "AdvanceWidths"); _advanceWidths = value; } } /// /// Array of glyph offsets. Added to the nominal glyph origin calculated above to generate the final origin for the glyph. /// Base glyphs generally have a glyph offset of (0,0), combining glyphs generally have an offset /// that places them correctly on top of the nearest preceeding base glyph. /// [CLSCompliant(false)] [TypeConverter(typeof(PointIListConverter))] public IListGlyphOffsets { get { CheckInitialized(); return _glyphOffsets; } set { CheckInitializing(); // This can only be set during initialization. // The list can be null, empty or non-empty list. // The consistency with other lists would be checked at EndInit() time. _glyphOffsets = value; } } /// /// Returns the language associated with the glyph run. /// public XmlLanguage Language { get { CheckInitialized(); return _language; } set { CheckInitializing(); // This can only be set during initialization. _language = value; } } ////// Identifies a specific device font for which the GlyphRun has been optimized. When a GlyphRun is /// being rendered on a device that has built-in support for this named font, then the GlyphRun should be rendered using a /// possibly device specific mechanism for selecting that font, and by sending the Unicode codepoints rather than the /// glyph indices. When rendering onto a device that does not include built-in support for the named font, /// this property should be ignored. /// public string DeviceFontName { get { CheckInitialized(); return _deviceFontName; } set { CheckInitializing(); // This can only be set during initialization. _deviceFontName = value; } } #endregion Public Properties ////// Glyph offsets /// The array is indexed starting with InitialGlyph /// internal Point GetGlyphOffset(int i) { if (_glyphOffsets == null || _glyphOffsets.Count == 0) return new Point(0, 0); return _glyphOffsets[i]; } ////// Returns whether GlyphRun contains anything to be drawn. /// Examples of GlyphRun's that don't contain ink are: /// - GlyphRun containing only spaces (" ") /// - GlyphRun wiithout any glyphs in it /// ///true if GlyphRun contains ink, false if it doesn't. private bool HasInk { get { if ((_flags & GlyphRunFlags.HasInkIsCached) != 0) { // // We've already calculated this property before // -- we just need to retrieve the cached bit. // return (_flags & GlyphRunFlags.HasInk) != 0; } else { // // Check if the glyph run contains anything to be drawn. // Cache the result and mark the property as cached. // bool hasInk = _glyphIndices.Count != 0 && !ComputeInkBoundingBox().IsEmpty; _flags |= (hasInk ? GlyphRunFlags.HasInk : GlyphRunFlags.None) | GlyphRunFlags.HasInkIsCached; return hasInk; } } } internal int GlyphCount { get { return _glyphIndices.Count; } } internal int CodepointCount { get { if (_characters != null && _characters.Count != 0) return _characters.Count; if (_clusterMap != null && _clusterMap.Count != 0) return _clusterMap.Count; return _glyphIndices.Count; } } #region Drawing and measurements ////// Computes ink bounding box for the glyph run. /// The rectangle is relative to the glyph run origin. /// ///The ink bounding box of the glyph run public Rect ComputeInkBoundingBox() { CheckInitialized(); // This can only be called on fully initialized GlyphRun if ((_flags & GlyphRunFlags.CacheInkBounds) != 0) { if (_inkBoundingBox != null) { return (Rect) _inkBoundingBox; } } bool italicSimulation = (_glyphTypeface.StyleSimulations & StyleSimulations.ItalicSimulation) != 0; // Special casing Left to Right layout with no italics allows an implementation that is // 12 times faster than the general case. Other combinations of Left to Right and // sideways layout also presents optimization opportunities that need to be implemented. // Italics is used infrequently, so adding the additional 8 routines necessary to handle italics // in combination with the other 4 routines is not justified. if (IsLeftToRight && !IsSideways && !italicSimulation) { Rect rect = ComputeInkBoundingBoxLtoR(); if ((_flags & GlyphRunFlags.CacheInkBounds) != 0) { _inkBoundingBox = rect; } return rect; } double accAdvance = 0; // We don't use Rect and Rect.Union to accumulate the bounding box // because this function is a hot spot and Rect methods perform extra checks that we don't need. double accLeft = double.PositiveInfinity; double accTop = double.PositiveInfinity; double accRight = double.NegativeInfinity; double accBottom = double.NegativeInfinity; for (int i = 0; i < GlyphCount; ++i) { double aw, ah, lsb, rsb, tsb, bsb, baseline; _glyphTypeface.GetGlyphMetrics( _glyphIndices[i], _renderingEmSize, out aw, out ah, out lsb, out rsb, out tsb, out bsb, out baseline ); Point glyphOffset = GetGlyphOffset(i); double originX; if (IsLeftToRight) { originX = accAdvance + glyphOffset.X; } else { // no languages support sideways and right to left in the same run Debug.Assert(!IsSideways); originX = -accAdvance - (aw + glyphOffset.X); } accAdvance += _advanceWidths[i]; double horBaselineOriginY = -glyphOffset.Y; double left, right, bottom, top; if (IsSideways) { horBaselineOriginY += aw / 2.0; bottom = horBaselineOriginY - lsb; top = horBaselineOriginY - aw + rsb; left = originX + tsb; right = left + ah - tsb - bsb; } else { left = originX + lsb; right = originX + aw - rsb; bottom = horBaselineOriginY + baseline; top = bottom - ah + tsb + bsb; } // extend the ink box for the glyph to account for italic simulation if (italicSimulation) { // the italic simulation is always applied relative to horizontal baseline right += Sin20 * (horBaselineOriginY - top); left -= Sin20 * (bottom - horBaselineOriginY); } // skip blank glyphs, as they don't contain ink if (left + InkMetricsEpsilon >= right || top + InkMetricsEpsilon >= bottom) continue; if (accLeft > left) accLeft = left; if (accTop > top) accTop = top; if (accRight < right) accRight = right; if (accBottom < bottom) accBottom = bottom; } Rect bounds; if (accLeft > accRight) { bounds = Rect.Empty; } else { bounds = new Rect( accLeft, accTop, accRight - accLeft, accBottom - accTop ); } if ((_flags & GlyphRunFlags.CacheInkBounds) != 0) { _inkBoundingBox = bounds; } return bounds; } private Rect ComputeInkBoundingBoxLtoR() { // We don't use Rect and Rect.Union to accumulate the bounding box // because this function is a hot spot and Rect methods perform extra checks that we don't need. double accLeft = double.PositiveInfinity; double accTop = double.PositiveInfinity; double accRight = double.NegativeInfinity; double accBottom = double.NegativeInfinity; double accAdvance = 0; int glyphCount = GlyphCount; if (GlyphOffsets != null) { for (int i = 0; i < glyphCount; ++i) { double aw, ah, lsb, rsb, tsb, bsb, baseline; _glyphTypeface.GetGlyphMetrics( _glyphIndices[i], _renderingEmSize, out aw, out ah, out lsb, out rsb, out tsb, out bsb, out baseline ); Point glyphOffset = GetGlyphOffset(i); double originX = accAdvance + glyphOffset.X; accAdvance += _advanceWidths[i]; double horBaselineOriginY = -glyphOffset.Y; double left, right, bottom, top; left = originX + lsb; right = originX + aw - rsb; bottom = horBaselineOriginY + baseline; top = bottom - ah + tsb + bsb; // skip blank glyphs, as they don't contain ink if (left + InkMetricsEpsilon >= right || top + InkMetricsEpsilon >= bottom) continue; if (accLeft > left) accLeft = left; if (accTop > top) accTop = top; if (accRight < right) accRight = right; if (accBottom < bottom) accBottom = bottom; } } else { for (int i = 0; i < glyphCount; ++i) { double aw, ah, lsb, rsb, tsb, bsb, baseline; _glyphTypeface.GetGlyphMetrics( _glyphIndices[i], _renderingEmSize, out aw, out ah, out lsb, out rsb, out tsb, out bsb, out baseline ); double left, right, top; left = accAdvance + lsb; right = accAdvance + aw - rsb; top = baseline - ah + tsb + bsb; accAdvance += _advanceWidths[i]; // skip blank glyphs, as they don't contain ink if (left + InkMetricsEpsilon >= right || top + InkMetricsEpsilon >= baseline) continue; if (accLeft > left) accLeft = left; if (accTop > top) accTop = top; if (accRight < right) accRight = right; if (accBottom < baseline) accBottom = baseline; } } if (accLeft > accRight) return Rect.Empty; return new Rect( accLeft, accTop, accRight - accLeft, accBottom - accTop ); } ////// Obtains geometry for the glyph run. /// ///The geometry returned contains the combined geometry of all glyphs in the glyph run. /// Overlapping contours are merged by performing a Boolean union operation. public Geometry BuildGeometry() { CheckInitialized(); // This can only be called on fully initialized GlyphRun GeometryGroup accumulatedGeometry = null; double accAdvance = 0; for (int i = 0; i < GlyphCount; ++i) { ushort glyphIndex = _glyphIndices[i]; double originX; if (IsLeftToRight) { originX = accAdvance; originX += GetGlyphOffset(i).X; } else { // no languages support sideways and right to left in the same run Debug.Assert(!IsSideways); double nominalAdvance = _glyphTypeface.GetAdvanceWidth(glyphIndex) * _renderingEmSize; originX = -accAdvance; originX -= (nominalAdvance + GetGlyphOffset(i).X); } accAdvance += _advanceWidths[i]; double originY = -GetGlyphOffset(i).Y; Geometry glyphGeometry = _glyphTypeface.ComputeGlyphOutline(glyphIndex, IsSideways, _renderingEmSize); if (glyphGeometry.IsEmpty()) continue; // transform glyphGeometry to the glyph origin glyphGeometry.Transform = new TranslateTransform(originX + _baselineOrigin.X, originY + _baselineOrigin.Y); if (accumulatedGeometry == null) { accumulatedGeometry = new GeometryGroup(); accumulatedGeometry.FillRule = FillRule.Nonzero; } accumulatedGeometry.Children.Add(glyphGeometry.GetOutlinedPathGeometry(RelativeFlatteningTolerance, ToleranceType.Relative)); } // Make sure to always return Geometry.Empty from public methods for empty geometries. if (accumulatedGeometry == null || accumulatedGeometry.IsEmpty()) return Geometry.Empty; return accumulatedGeometry; } ////// Computes the alignment box for the glyph run. /// The alignment box is relative to origin. /// ///The alignment box for the glyph run. public Rect ComputeAlignmentBox() { CheckInitialized(); // This can only be called on fully initialized GlyphRun if (IsLeftToRight) { return new Rect( 0, -Ascent, AdvanceWidth, Height ); } else { // cache AdvanceWidth value in a local variable because it involves a loop double advanceWidth = AdvanceWidth; return new Rect( -advanceWidth, -Ascent, advanceWidth, Height ); } } ////// Temporary helper to draw a glyph run background. /// We hope to remove all uses of it, as fundamentally this is not the right way /// to handle background drawing. /// internal void EmitBackground(DrawingContext dc, Brush backgroundBrush) { if (backgroundBrush != null) { Rect backgroundRect; if (IsLeftToRight) { backgroundRect = new Rect( _baselineOrigin.X, _baselineOrigin.Y - Ascent, AdvanceWidth, Height ); } else { backgroundRect = new Rect( _baselineOrigin.X - AdvanceWidth, _baselineOrigin.Y - Ascent, AdvanceWidth, Height ); } dc.DrawRectangle( backgroundBrush, null, backgroundRect ); } } #endregion Drawing and measurements #region DUCE.IResource implementation ////// A structure to keep two scaling ratios fetched from given Matrix. /// internal struct Scale { internal Scale(ref Matrix matrix) { double m11 = matrix.M11; double m12 = matrix.M12; double m21 = matrix.M21; double m22 = matrix.M22; // Calculate redundant data. _baseVectorX = Math.Sqrt(m11 * m11 + m12 * m12); // Check for wrong matrix. if (DoubleUtil.IsNaN(_baseVectorX)) _baseVectorX = 0; _baseVectorY = _baseVectorX == 0 ? 0 : Math.Abs(m11 * m22 - m12 * m21) / _baseVectorX; if (DoubleUtil.IsNaN(_baseVectorY)) _baseVectorY = 0; } internal bool IsValid { get { return _baseVectorX != 0 && _baseVectorY != 0; } } internal bool IsSame(ref Scale scale) { // // allow some imprecision that can appear because // of matrix computations. // return _baseVectorX * 0.999999999 <= scale._baseVectorX && _baseVectorX * 1.000000001 >= scale._baseVectorX && _baseVectorY * 0.999999999 <= scale._baseVectorY && _baseVectorY * 1.000000001 >= scale._baseVectorY; } internal double _baseVectorX, _baseVectorY; } private DUCE.MultiChannelResource _mcr = new DUCE.MultiChannelResource(); ////// Generate a series of requests to create or update /// slave glyph run resource and all depending data. /// DUCE.ResourceHandle DUCE.IResource.AddRefOnChannel(DUCE.Channel channel) { CheckInitialized(); // This can only be called on fully initialized GlyphRun using (CompositionEngineLock.Acquire()) { if (_mcr.CreateOrAddRefOnChannel(channel, DUCE.ResourceType.TYPE_GLYPHRUN)) { CreateOnChannel(channel); } return _mcr.GetHandle(channel); } } ////// Generates request to delete slave glyph run resource. /// void DUCE.IResource.ReleaseOnChannel(DUCE.Channel channel) { CheckInitialized(); // This can only be called on fully initialized GlyphRun using (CompositionEngineLock.Acquire()) { _mcr.ReleaseOnChannel(channel); } } ////// This is only implemented by Visual and Visual3D. /// void DUCE.IResource.RemoveChildFromParent(DUCE.IResource parent, DUCE.Channel channel) { throw new NotImplementedException(); } ////// This is only implemented by Visual and Visual3D. /// DUCE.ResourceHandle DUCE.IResource.Get3DHandle(DUCE.Channel channel) { throw new NotImplementedException(); } ////// Returns current resource handle, allocated recently by AddRefOnChannel. /// DUCE.ResourceHandle DUCE.IResource.GetHandle(DUCE.Channel channel) { CheckInitialized(); // This can only be called on fully initialized GlyphRun return _mcr.GetHandle(channel); } int DUCE.IResource.GetChannelCount() { return _mcr.GetChannelCount(); } DUCE.Channel DUCE.IResource.GetChannel(int index) { return _mcr.GetChannel(index); } ////// Send to channel command sequence to create slave resource. /// ////// Critical: This code acceses an unsafe code block /// TreatAsSafe: This does not expose any data and the unsafe code blocks /// needs to be verified for correctness /// [SecurityCritical,SecurityTreatAsSafe] private void CreateOnChannel(DUCE.Channel channel) { Debug.Assert(_glyphTypeface != null); int glyphCount = GlyphCount; string fontFileName = _glyphTypeface.FontSource.GetUriString(); // // The InkBoundingBox + the Origin produce the true InkBoundingBox. // // Not sure why the bounding box code doesn't adjust for this when you // ask for the bounding box, instead everything // that is interested in the bounding box has to do this calculation. // Rect adjustedInkBoundingBox = ComputeInkBoundingBox(); if (!adjustedInkBoundingBox.IsEmpty) { adjustedInkBoundingBox.Offset((Vector)BaselineOrigin); } DUCE.MILCMD_GLYPHRUN_CREATE command; command.Type = MILCMD.MilCmdGlyphRunCreate; command.Handle = _mcr.GetHandle(channel); command.hGlyphCache = channel.GlyphCache.Handle; command.FontFaceIndex = _glyphTypeface.FaceIndex; command.GlyphRunFlags = ComposeFlags(); command.Origin.X = (float)_baselineOrigin.X; command.Origin.Y = (float)_baselineOrigin.Y; command.MuSize = (float)_renderingEmSize; command.ManagedBounds = (Rect)adjustedInkBoundingBox; command.FontFileNameLength = checked((UInt16)(fontFileName.Length + 1)); command.GlyphCount = checked((UInt16)glyphCount); unsafe { // calculate variable data size int varDataSize = (fontFileName.Length + 1) * sizeof(char); // '\0' terminating char if (_glyphOffsets != null && _glyphOffsets.Count != 0) { varDataSize += glyphCount * (2 * sizeof(float)); // positionXY } else { varDataSize += (glyphCount - 1) * sizeof(float); // positionX } varDataSize += glyphCount * sizeof(ushort); // glyph indices channel.BeginCommand( (byte*)&command, sizeof(DUCE.MILCMD_GLYPHRUN_CREATE), varDataSize ); { // Copy the font filename string char* pFontFileName = stackalloc char[fontFileName.Length + 1]; for (int i = 0; i < fontFileName.Length; i++) { pFontFileName[i] = fontFileName[i]; } pFontFileName[fontFileName.Length] = '\0'; channel.AppendCommandData((byte*)pFontFileName, (fontFileName.Length + 1) * sizeof(char)); // fill variable data block double rcmuSize = 1.0 / _renderingEmSize; if (Double.IsInfinity(rcmuSize)) { // Protect against extremely small _renderingEmSize: // denormalized numbers like 5E-324 can cause overflow on // calculating reciprocal value. rcmuSize = 0; } double xRatio = ((command.GlyphRunFlags & (uint)MilGlyphRun.IsLeftToRight) != 0) ? rcmuSize : -rcmuSize; if (_glyphOffsets != null && _glyphOffsets.Count != 0) { double yRatio = -rcmuSize; double accAdvances = 0; if (glyphCount <= MaxStackAlloc / (2 * sizeof(float))) { // glyph count small enough, send all data at once float* pGlyphPositions = stackalloc float[glyphCount * 2]; for (int i = 0; i < glyphCount; i++) { double x = _glyphOffsets[i].X + accAdvances; double y = _glyphOffsets[i].Y; accAdvances += _advanceWidths[i]; pGlyphPositions[2 * i] = (float)(x * xRatio); pGlyphPositions[2 * i + 1] = (float)(y * yRatio); } channel.AppendCommandData((byte*)pGlyphPositions, glyphCount * (2 * sizeof(float))); } else { // glyph count is not small, use per-glyph transmitting float* pGlyphPositions = stackalloc float[2]; for (int i = 0; i < glyphCount; i++) { double x = _glyphOffsets[i].X + accAdvances; double y = _glyphOffsets[i].Y; accAdvances += _advanceWidths[i]; pGlyphPositions[0] = (float)(x * xRatio); pGlyphPositions[1] = (float)(y * yRatio); channel.AppendCommandData((byte*)pGlyphPositions, 2 * sizeof(float)); } } } else if (glyphCount > 1) { // accumulate advance widths and convert them to "glyph space" double accAdvances = 0; if (glyphCount <= MaxStackAlloc / sizeof(float)) { // glyph count small enough, send all data at once float* pPositionX = stackalloc float[glyphCount - 1]; // skipping first glyph position that's always zero for (int i = 0; i < (glyphCount - 1); ++i) { accAdvances += _advanceWidths[i]; pPositionX[i] = (float)(accAdvances * xRatio); } channel.AppendCommandData((byte*)pPositionX, (glyphCount - 1) * sizeof(float)); } else { // glyph count is not small, use per-glyph transmitting for (int i = 0; i < (glyphCount - 1); ++i) { accAdvances += _advanceWidths[i]; float positionX = (float)(accAdvances * xRatio); channel.AppendCommandData((byte*)&positionX, sizeof(float)); } } } { // transmit glyph indices if (glyphCount <= MaxStackAlloc / sizeof(ushort)) { // glyph count small enough, send all data at once ushort* pGlyphIndices = stackalloc ushort[glyphCount]; for (int i = 0; i < glyphCount; ++i) { pGlyphIndices[i] = _glyphIndices[i]; } channel.AppendCommandData((byte*)pGlyphIndices, glyphCount * sizeof(ushort)); } else { // glyph count is not small, use per-glyph transmitting for (int i = 0; i < glyphCount; ++i) { ushort glyphIndex = _glyphIndices[i]; channel.AppendCommandData((byte*)&glyphIndex, sizeof(ushort)); } } } } channel.EndCommand(); } } ////// Gather flags that affect: /// - glyph run rendering /// - glyph rasterization /// - the way how glyph run data is packed /// private UInt16 ComposeFlags() { UInt16 flags = 0; if (_glyphTypeface.FontTechnology != FontTechnology.PostscriptOpenType) { flags |= (UInt16)MilGlyphRun.IsTrueType; FontFaceLayoutInfo.RenderingHints renderingHints = _glyphTypeface.RenderingHints; // When dealing with East Asian fonts containing embedded bitmaps // we apply a special 6x5 mode with hinting at em square and enhanced constract algorithms. if (renderingHints == FontFaceLayoutInfo.RenderingHints.LegacyEastAsian) { flags |= (UInt16)MilGlyphRun.ForceVAA | (UInt16)MilGlyphRun.VerticalDropOut | (UInt16)MilGlyphRun.OverContrast; } else { flags |= (UInt16)MilGlyphRun.Hinting; } } else { // For CFF fonts we always use 6x5 overscale hinted mode, because stem thickening happens in the CFF rasterizer. flags |= (UInt16)MilGlyphRun.Hinting | (UInt16)MilGlyphRun.ForceVAA; } StyleSimulations styleSimulations = _glyphTypeface.StyleSimulations; if ((styleSimulations & StyleSimulations.BoldSimulation) != 0) flags |= (UInt16)MilGlyphRun.BoldSimulation; if ((styleSimulations & StyleSimulations.ItalicSimulation) != 0) flags |= (UInt16)MilGlyphRun.ItalicSimulation; if (IsSideways) flags |= (UInt16)MilGlyphRun.Sideways; // if ((_bidiLevel & 1) == 0) flags |= (UInt16)MilGlyphRun.IsLeftToRight; if (_glyphOffsets != null && _glyphOffsets.Count != 0) flags |= (UInt16)MilGlyphRun.HasYPositions; // Encode font contrast adjustment into 3 bits masked by MilGlyphRunPrecontrastLevel. UInt16 fontContrastAdjustment = (UInt16)(_glyphTypeface.FontContrastAdjustment + (int)MilGlyphRun.PrecontrastOffset); // If the resulting value spills outside the mask, this indicates a code bug in the GlyphTypeface code. Debug.Assert(((fontContrastAdjustment << (int)MilGlyphRun.PrecontrastShift) & ~(UInt16)MilGlyphRun.PrecontrastLevel) == 0); flags |= (UInt16)(fontContrastAdjustment << (int)MilGlyphRun.PrecontrastShift); return flags; } #endregion DUCE.IResource implementation #region Hit testing ////// Given a code point index in the caret stop array, finds the nearest pair of caret stops. /// /// Character index to start the search from. Doesn't have to be snapped. /// GlyphRun CaretStops array. Guaranteed to be non-null. /// Nearest caret stop index, or -1 if there are no caret stops. /// Code points until the next caret stop, or -1 if there is no next caret stop. private void FindNearestCaretStop( int characterIndex, IListcaretStops, out int caretStopIndex, out int codePointsUntilNextStop) { caretStopIndex = -1; codePointsUntilNextStop = -1; if (characterIndex < 0 || characterIndex >= caretStops.Count) return; // Find the closest caret stop at the character index or to the left of it. for (int i = characterIndex; i >= 0; --i) { if (caretStops[i]) { caretStopIndex = i; break; } } // Couldn't find a caret stop at the character index or to the left of it. // Search to the right. if (caretStopIndex == -1) { for (int i = characterIndex + 1; i < caretStops.Count; ++i) { if (caretStops[i]) { caretStopIndex = i; break; } } } // No caret stops found, the glyph run is not hit testable. if (caretStopIndex == -1) { return; } for (int lastStop = caretStopIndex + 1; lastStop < caretStops.Count; ++lastStop) { if (caretStops[lastStop]) { // There is a next caret stop. codePointsUntilNextStop = lastStop - caretStopIndex; return; } } // There is no next caret stop. } /// /// This class implements behavior of a Boolean list that contains all true values. /// This allows us to have a single code path in hit testing API. /// private class DefaultCaretStopList : IList{ public DefaultCaretStopList(int codePointCount) { _count = codePointCount + 1; } #region IList Members public int IndexOf(bool item) { throw new NotSupportedException(); } public void Insert(int index, bool item) { throw new NotSupportedException(); } public bool this[int index] { get { return true; } set { throw new NotSupportedException(); } } public void RemoveAt(int index) { throw new NotSupportedException(); } #endregion #region ICollection Members public void Add(bool item) { throw new NotSupportedException(); } public void Clear() { throw new NotSupportedException(); } public bool Contains(bool item) { throw new NotSupportedException(); } public void CopyTo(bool[] array, int arrayIndex) { throw new NotSupportedException(); } public int Count { get { return _count; } } public bool IsReadOnly { get { return true; } } public bool Remove(bool item) { throw new NotSupportedException(); } #endregion #region IEnumerable Members IEnumerator IEnumerable .GetEnumerator() { throw new NotSupportedException(); } #endregion #region IEnumerable Members IEnumerator IEnumerable.GetEnumerator() { throw new NotSupportedException(); } #endregion private int _count; } /// /// This class implements behavior of a 1:1 cluster map. /// This allows us to have a single code path in hit testing API. /// private class DefaultClusterMap : IList{ public DefaultClusterMap(int count) { _count = count; } #region IList Members public int IndexOf(ushort item) { throw new NotSupportedException(); } public void Insert(int index, ushort item) { throw new NotSupportedException(); } public ushort this[int index] { get { return (ushort)index; } set { throw new NotSupportedException(); } } public void RemoveAt(int index) { throw new NotSupportedException(); } #endregion #region ICollection Members public void Add(ushort item) { throw new NotSupportedException(); } public void Clear() { throw new NotSupportedException(); } public bool Contains(ushort item) { throw new NotSupportedException(); } public void CopyTo(ushort[] array, int arrayIndex) { throw new NotSupportedException(); } public int Count { get { return _count; } } public bool IsReadOnly { get { return true; } } public bool Remove(ushort item) { throw new NotSupportedException(); } #endregion #region IEnumerable Members IEnumerator IEnumerable .GetEnumerator() { throw new NotSupportedException(); } #endregion #region IEnumerable Members IEnumerator IEnumerable.GetEnumerator() { throw new NotSupportedException(); } #endregion private int _count; } #endregion Hit testing #region ISupportInitialize interface for Xaml serialization void ISupportInitialize.BeginInit() { if (IsInitialized) { // Cannot initialize a GlyphRun that is completely initialized. throw new InvalidOperationException(SR.Get(SRID.OnlyOneInitialization)); } if (IsInitializing) { // Cannot initialize a GlyphRun that is already being initialized. throw new InvalidOperationException(SR.Get(SRID.InInitialization)); } IsInitializing = true; } void ISupportInitialize.EndInit() { if (!IsInitializing) { // Cannot EndInit a GlyphRun that is not being initialized. throw new InvalidOperationException(SR.Get(SRID.NotInInitialization)); } // // Fully initilize the GlyphRun. The method will check for consistency // between all the properties. // Initialize( _glyphTypeface, _bidiLevel, (_flags & GlyphRunFlags.IsSideways) != 0, _renderingEmSize, _glyphIndices, _baselineOrigin, (_advanceWidths == null ? null : new ThousandthOfEmRealDoubles(_renderingEmSize, _advanceWidths)), (_glyphOffsets == null ? null : new ThousandthOfEmRealPoints(_renderingEmSize, _glyphOffsets)), _characters, _deviceFontName, _clusterMap, _caretStops, _language, true // throwIfOverflow ); // User should be able to fix errors that are only caught at EndInit() time. So set Initializing flag to // false after Initialization succeeds. IsInitializing = false; } private void CheckInitialized() { if (!IsInitialized) { throw new InvalidOperationException(SR.Get(SRID.InitializationIncomplete)); } // Ensure the bits are set consistently. The object cannot be in both states. Debug.Assert(!IsInitializing); } private void CheckInitializing() { if (!IsInitializing) { throw new InvalidOperationException(SR.Get(SRID.NotInInitialization)); } // Ensure the bits are set consistently. The object cannot be in both states. Debug.Assert(!IsInitialized); } private bool IsInitializing { get { return (_flags & GlyphRunFlags.IsInitializing) != 0; } set { if (value) { _flags |= GlyphRunFlags.IsInitializing; } else { _flags &= (~GlyphRunFlags.IsInitializing); } } } private bool IsInitialized { get { return (_flags & GlyphRunFlags.IsInitialized) != 0; } set { if (value) { _flags |= GlyphRunFlags.IsInitialized; } else { _flags &= (~GlyphRunFlags.IsInitialized); } } } #endregion //----------------------------------------------------- // // Private Enumerations // //------------------------------------------------------ #region Private Enumerations /// /// Glyph run flags. /// [Flags] private enum GlyphRunFlags : byte { ////// No flags set. /// It also represents the state in which the GlyphRun has not been initialized. /// At this state, all operations on the object would cause InvalidOperationException. /// The object can only transit to 'IsInitializing' state with BeginInit() call. /// None = 0x00, ////// Set to display the GlyphRun sideways. /// IsSideways = 0x01, ////// Set if the glyph run contains anything to be drawn. /// HasInk = 0x02, ////// Set if the HasInk bit above has already been calculated and cached. /// HasInkIsCached = 0x04, ////// The state in which the GlyphRun object is fully initialized. At this state the object /// is fully functional. There is no valid transition out of the state. /// IsInitialized = 0x08, ////// The state in which the GlyphRun is being initialized. At this state, user can /// set values into the required properties. The object can only transit to 'IsInitialized' state /// with EndInit() call. /// IsInitializing = 0x10, ////// Caching ink bounds /// CacheInkBounds = 0x20, } #endregion Private Enumerations //----------------------------------------------------- // // Private Fields // //----------------------------------------------------- #region Private Fields private Point _baselineOrigin; private GlyphRunFlags _flags; private double _renderingEmSize; private IList_glyphIndices; private IList _advanceWidths; private IList _glyphOffsets; private int _bidiLevel; private GlyphTypeface _glyphTypeface; private IList _characters; private IList _clusterMap; private IList _caretStops; private XmlLanguage _language; private string _deviceFontName; private object _inkBoundingBox; // Used when CacheInkBounds is on // the sine of 20 degrees private const double Sin20 = 0.34202014332566873304409961468226; // This is the precision that is used to decide that glyph metrics are equal, // for example when detecting blank glyphs. // The chosen value is greater than typical floating point precision loss // but smaller than typical design font unit (1/1024th or 1/2048th). private const double InkMetricsEpsilon = 0.0000001; // Dummy font hinting size private const double DefaultFontHintingSize = 12.0; // Tolerance for flattening Bezier curves when calling GetOutlinedPathGeometry. internal static double RelativeFlatteningTolerance = 0.01; // The constants that delimit glyph run size. // Should correspond to unmanaged ones in GlyphRunCore.h. internal const int MaxGlyphCount = 0xFFFF; internal const int OverscaledCoordMax = 0xFFFFF; internal const int GeometryThreshold = 100; internal const int MaxOvescale = 8; internal const int MaxOverscaledBitsInGlyphRun = 800000000; // 800 MBits = 100 MBytes internal const int MaxStackAlloc = 1024; #endregion Private Fields } } // File provided for Reference Use Only by Microsoft Corporation (c) 2007. //---------------------------------------------------------------------------- // // Copyright (c) Microsoft Corporation. All rights reserved. // // Description: The GlyphRun object represents a sequence of glyphs from a single face of a single font at a single size, // and with a single rendering style. // // See specs at // http://team/sites/Avalon/Specs/Glyphs%20element%20and%20GlyphRun%20object.htm // http://team/sites/Avalon/Specs/Glyph%20Run%20hit%20testing%20and%20caret%20placement%20API.htm // // // History: // 02/18/2003 : mleonov - Created // //--------------------------------------------------------------------------- // Enable presharp pragma warning suppress directives. #pragma warning disable 1634, 1691 using System; using System.Collections; using System.Collections.Generic; using System.ComponentModel; using System.Diagnostics; using System.Globalization; using System.Text; using System.Windows; using System.Windows.Media; using System.Windows.Media.Converters; using System.Windows.Media.Composition; using System.Windows.Media.TextFormatting; using System.Windows.Markup; using System.Runtime.InteropServices; using MS.Internal; using MS.Internal.FontCache; using MS.Internal.FontFace; using MS.Internal.TextFormatting; using MS.Utility; using System.Security; using System.Security.Permissions; using SR=MS.Internal.PresentationCore.SR; using SRID=MS.Internal.PresentationCore.SRID; namespace System.Windows.Media { /// /// The GlyphRun object represents a sequence of glyphs from a single face of a single font at a single size, /// and with a single rendering style. /// // public class GlyphRun : DUCE.IResource, ISupportInitialize { //----------------------------------------------------- // // Constructors // //----------------------------------------------------- #region Constructors ////// Construct an uninitialized GlyphRun object. Caller should call ISupportInitialize.BeginInit() /// to begin initialization and call ISupportInitialize.EndInit() to finish the initialization. /// The GlyphRun does not support all the operations until it is fully initialized. /// public GlyphRun() { } ////// Constructs a new GlyphRun object. /// /// GlyphTypeface of the GlyphRun object /// Bidi level of the GlyphRun object /// Set to true to display the GlyphRun sideways /// Font rendering size in drawing surface units (96ths of an inch). /// The list of font indices that represent glyphs in this run. /// Origin of the first glyph in the run. /// The glyph is placed so that the leading edge of its advance vector /// and its baseline intersect this point. /// /// The list of advance widths, one for each glyph in GlyphIndices. /// The nominal origin of the nth glyph (n > 0) in the run is the nominal origin /// of the n-1th glyph plus the n-1th advance width added along the runs advance vector. /// Base glyphs generally have a non-zero advance width, combining glyphs generally have a zero advance width. /// /// The list of glyph offsets. Added to the nominal glyph origin calculated above to generate the final origin for the glyph. /// Base glyphs generally have a glyph offset of (0,0), combining glyphs generally have an offset /// that places them correctly on top of the nearest preceeding base glyph. /// /// Characters represented by this glyphrun /// /// Identifies a specific device font for which the GlyphRun has been optimized. When a GlyphRun is /// being rendered on a device that has built-in support for this named font, then the GlyphRun should be rendered using a /// possibly device specific mechanism for selecting that font, and by sending the Unicode codepoints rather than the /// glyph indices. When rendering onto a device that does not include built-in support for the named font, /// this property should be ignored. /// /// The list that maps characters in the glyph run to glyph indices. /// There is one entry per character in Characters list. /// Each value gives the offset of the first glyph in GlyphIndices /// that represents the corresponding character in Characters. /// Where multiple characters map to a single glyph, or to a glyph group /// that cannot be broken down to map exactly to individual characters, /// the entries for all the characters have the same value: /// the offset of the first glyph that represents this group of characters. /// If the list is null or empty, sequential 1 to 1 mapping is assumed. /// /// A list of caret stops for the glyphs /// Language of the GlyphRun [CLSCompliant(false)] public GlyphRun( GlyphTypeface glyphTypeface, int bidiLevel, bool isSideways, double renderingEmSize, IListglyphIndices, Point baselineOrigin, IList advanceWidths, IList glyphOffsets, IList characters, string deviceFontName, IList clusterMap, IList caretStops, XmlLanguage language ) { // Suppress PRESharp warning that glyphIndices and advanceWidths are not validated and can be null. // They can indeed be null, but that's perfectly OK. An explicit null check in the constructor is // not required. #pragma warning suppress 56506 Initialize( glyphTypeface, bidiLevel, isSideways, renderingEmSize, glyphIndices, baselineOrigin, advanceWidths, glyphOffsets, characters, deviceFontName, clusterMap, caretStops, language, true // throwIfOverflow ); // GlyphRunFlags.CacheInkBounds enanbles ink bounding box caching. Bounding box caching would cost // 32 bytes per GlyphRun. We do not want to enable it for all cases possible working set increase. // For Line layout, ink bounding box is only used a few times, so caching is disabled because it will // go through TryCreate below. Memory cost: 1 pointer. // For loading XPS in which bounding box calculation are called a lot in hit testing, Glyphs.cs will // call this constructor, which enables caching. Memory cost: 1 pointer + boxed Rect. // If we late decide it's worthwhile to cache for all, memory cost can be reduced to one Rect (32-bytes). // If we decide single precision is good enough, it can be reduced to 16 bytes. _flags |= GlyphRunFlags.CacheInkBounds; } /// /// Creates a new GlyphRun object. This method is similar to the constructor with /// the same argument list except that it returns null instead of throwing an /// exception if the GlyphRun area or a coordinate exceed the maximum value. /// internal static GlyphRun TryCreate( GlyphTypeface glyphTypeface, int bidiLevel, bool isSideways, double renderingEmSize, IListglyphIndices, Point baselineOrigin, IList advanceWidths, IList glyphOffsets, IList characters, string deviceFontName, IList clusterMap, IList caretStops, XmlLanguage language ) { GlyphRun glyphRun = new GlyphRun(); // Suppress PRESharp warning that glyphIndices and advanceWidths are not validated and can be null. // They can indeed be null, but that's perfectly OK. An explicit null check in the constructor is // not required. #pragma warning suppress 56506 glyphRun.Initialize( glyphTypeface, bidiLevel, isSideways, renderingEmSize, glyphIndices, baselineOrigin, advanceWidths, glyphOffsets, characters, deviceFontName, clusterMap, caretStops, language, false // throwIfOverflow ); // Cached GlyphRun bounds are needed to pass to the render thread glyphRun._flags |= GlyphRunFlags.CacheInkBounds; if (glyphRun.IsInitialized) return glyphRun; else return null; } private void Initialize( GlyphTypeface glyphTypeface, int bidiLevel, bool isSideways, double renderingEmSize, IList glyphIndices, Point baselineOrigin, IList advanceWidths, IList glyphOffsets, IList characters, string deviceFontName, IList clusterMap, IList caretStops, XmlLanguage language, bool throwOnOverflow ) { // The default branch prediction rules for modern processors specify that forward branches // are not to be taken. If the branch is in fact taken, all of the speculatively executed code // must be discarded, the processor pipeline flushed, and then reloaded. This results in a // processor stall of at least 42 cycles for the P4 Northwood for each mis-predicted branch. // The deeper the processor pipeline the higher the cost, i.e. Prescott processors. // Checking for multiple incorrect parameters in a method with high call count like this one can // easily add significant overhead for no reason. Note that the C# compiler should be able to make // reasonable assumptions about branches that throw exceptions, but the current whidbey // implemenation is weak in this regard. Also the current IBC tools are unable to add branch // prediction hints to improve behavior based on run time information. Also note that adding // branch prediction hints increases code size by a byte per branch and doing this in every // method that is coded without default branch prediction behavior in mind would add an // unacceptable amount of working set. if ((glyphTypeface != null) && (glyphIndices != null) && (advanceWidths != null) && (renderingEmSize >= 0.0) && (glyphIndices.Count > 0) && (glyphIndices.Count <= MaxGlyphCount) && (advanceWidths.Count == glyphIndices.Count) && ((glyphOffsets == null) || ((glyphOffsets != null) && (glyphOffsets.Count != 0) && (glyphOffsets.Count == glyphIndices.Count)))) { // Set member variables here, // so that GlyphRun properties can be calculated in advanced validation code. _glyphIndices = glyphIndices; _characters = characters; _clusterMap = clusterMap; _baselineOrigin = baselineOrigin; _renderingEmSize = renderingEmSize; _advanceWidths = advanceWidths; _glyphOffsets = glyphOffsets; _glyphTypeface = glyphTypeface; _flags = (isSideways ? GlyphRunFlags.IsSideways : GlyphRunFlags.None); _bidiLevel = bidiLevel; _caretStops = caretStops; _language = language; _deviceFontName = deviceFontName; if (characters != null && characters.Count != 0) { if (clusterMap != null && clusterMap.Count != 0) { if (clusterMap.Count == characters.Count) { // Perform some simple cluster map validation. // First entry should be zero, the entries should be monotonic and shouldn't point outside of the glyph indices range. if (clusterMap[0] == 0) { int glyphCount = GlyphCount; int mapCount = clusterMap.Count; ushort previous = clusterMap[0]; for (int i = 1; i < mapCount; ++i) { ushort current = clusterMap[i]; if ((current >= previous) && (current < glyphCount)) { previous = current; } else { if (clusterMap[i] < clusterMap[i - 1]) throw new ArgumentException(SR.Get(SRID.ClusterMapEntriesShouldNotDecrease), "clusterMap"); if (clusterMap[i] >= GlyphCount) throw new ArgumentException(SR.Get(SRID.ClusterMapEntryShouldPointWithinGlyphIndices), "clusterMap"); } } } else { throw new ArgumentException(SR.Get(SRID.ClusterMapFirstEntryMustBeZero), "clusterMap"); } } else { throw new ArgumentException(SR.Get(SRID.CollectionNumberOfElementsShouldBeEqualTo, characters.Count), "clusterMap"); } } else { if (GlyphCount != characters.Count) throw new ArgumentException(SR.Get(SRID.CollectionNumberOfElementsShouldBeEqualTo, GlyphCount), "clusterMap"); } } if (caretStops != null && caretStops.Count != 0) { if (caretStops.Count != CodepointCount + 1) throw new ArgumentException(SR.Get(SRID.CollectionNumberOfElementsShouldBeEqualTo, CodepointCount + 1), "caretStops"); } if (isSideways && (bidiLevel & 1) != 0) throw new ArgumentException(SR.Get(SRID.SidewaysRTLTextIsNotSupported)); { // Check given arguments against big numbers. // The purpose of this block is to ensure that we'll // never get arithmetic overflow on rendering, and reduce // the risk of memory overflow. // Too big numbers will be discovered right here and cause // throwing the exception. // Coordinates are submitted to rendering in normalized form, // i.e. divided by renderingEmSize. // On rendering, coordinate values are multiplied by rasterization scale // that is known not to exceed sc_uGeometryThreshold * sc_uMaxOvescale. // Obtained values may be shifted by Origin and Advance values taken from // GlyphBitmap. The latter two values are shifted 16-bit short. The sum is then // converted to integer value using CFloatFPU::SmallRound routine // that accepts numbers in range -OverscaledCoordMax <= N <= OverscaledCoordMax. // Hence we have available maximum for given coordinate values: double coordMax = renderingEmSize * ((double)(OverscaledCoordMax - 0x10000)) / (GeometryThreshold * MaxOvescale); double glyphXMin = 0.0; double glyphXMax = 0.0; double glyphYMin = 0.0; double glyphYMax = 0.0; // First, calculate the smallest rectangle containing anchor points // of all of the glyphs. if (glyphOffsets != null) { double glyphX = glyphOffsets[0].X; double glyphY = glyphOffsets[0].Y; glyphXMin = glyphX; glyphXMax = glyphX; glyphYMin = glyphY; glyphYMax = glyphY; if ((Math.Abs(glyphX) <= coordMax) && (Math.Abs(glyphY) <= coordMax)) { double positionX = 0; int glyphCount = GlyphCount; for (int i = 1; i < glyphCount; i++) { positionX += advanceWidths[i - 1]; if (Math.Abs(positionX) <= coordMax) { glyphX = positionX; glyphY = 0; double glyphOffsetX = glyphOffsets[i].X; double glyphOffsetY = glyphOffsets[i].Y; if (Math.Abs(glyphOffsetX) <= coordMax && (Math.Abs(glyphOffsetY) <= coordMax)) { glyphX += glyphOffsetX; glyphY += glyphOffsetY; if (glyphX < glyphXMin) glyphXMin = glyphX; else if (glyphX > glyphXMax) glyphXMax = glyphX; if (glyphY < glyphYMin) glyphYMin = glyphY; else if (glyphY > glyphYMax) glyphYMax = glyphY; // Continue looping, no error continue; } } // We will get here if either range check above fails if (throwOnOverflow) ReportCoordinateOverflow(i, renderingEmSize, coordMax); else return; } } else { if (throwOnOverflow) ReportCoordinateOverflow(0, renderingEmSize, coordMax); else return; } } else { double glyphX = 0.0; int glyphCount = GlyphCount; for (int i = 1; i < glyphCount; i++) { glyphX += advanceWidths[i - 1]; if (Math.Abs(glyphX) <= coordMax) { if (glyphX < glyphXMin) glyphXMin = glyphX; else if (glyphX > glyphXMax) glyphXMax = glyphX; } else if (throwOnOverflow) { ReportCoordinateOverflow(i, renderingEmSize, coordMax); } else { return; } } } // We have the rectangle bounds (glyphXMin, glyphXMax, glyphYMin, glyphYMax) // in local coordinate space. Area occupied by glyph run is bigger because // glyphs have sizes. Note that glyph anchor point might be out of glyph // bounding rectangle, due to overhang. We allow 2*renderingEmSize offset from // glyph anchor point to any of four lines that compose glyph bounding rectangle. // // Now we can estimate the area occupied by glyph run. double relativeWidth = (glyphXMax - glyphXMin) / renderingEmSize + 4; double relativeHeight = (glyphYMax - glyphYMin) / renderingEmSize + 4; double relativeArea = relativeWidth * relativeHeight; double scaleMax = GeometryThreshold * MaxOvescale; double relativeAreaMax = MaxOverscaledBitsInGlyphRun / (scaleMax * scaleMax); // Reject the glyph run that occupies too big area. if (relativeArea > relativeAreaMax) { if (throwOnOverflow) ReportAreaOverflow(relativeArea, relativeAreaMax); else return; } } } else { if (DoubleUtil.IsNaN(renderingEmSize)) throw new ArgumentOutOfRangeException("renderingEmSize", SR.Get(SRID.ParameterValueCannotBeNaN)); if (renderingEmSize < 0.0) throw new ArgumentOutOfRangeException("renderingEmSize", SR.Get(SRID.ParameterValueCannotBeNegative)); if (glyphTypeface == null) throw new ArgumentNullException("glyphTypeface"); if (glyphIndices == null) throw new ArgumentNullException("glyphIndices"); if (glyphIndices.Count <= 0) throw new ArgumentException(SR.Get(SRID.CollectionNumberOfElementsMustBeGreaterThanZero), "glyphIndices"); if (glyphIndices.Count > MaxGlyphCount) { throw new ArgumentException(SR.Get(SRID.CollectionNumberOfElementsMustBeLessOrEqualTo, MaxGlyphCount), "glyphIndices"); } if (advanceWidths == null) throw new ArgumentNullException("advanceWidths"); if (advanceWidths.Count != glyphIndices.Count) throw new ArgumentException(SR.Get(SRID.CollectionNumberOfElementsShouldBeEqualTo, glyphIndices.Count), "advanceWidths"); if (glyphOffsets != null && glyphOffsets.Count != 0 && glyphOffsets.Count != glyphIndices.Count) throw new ArgumentException(SR.Get(SRID.CollectionNumberOfElementsShouldBeEqualTo, glyphIndices.Count), "glyphOffsets"); // We should've caught all invalid cases above and thrown appropriate exceptions. Invariant.Assert(false); } IsInitialized = true; // The glyphrun is completely initialized } private void ReportCoordinateOverflow(int index, double renderingEmSize, double coordMax) { string message = SR.Get(SRID.GlyphCoordinateTooBig); message = string.Format(CultureInfo.CurrentCulture, message, index, renderingEmSize, coordMax); throw new OverflowException(message); } private void ReportAreaOverflow(double relativeArea, double relativeAreaMax) { string message = SR.Get(SRID.GlyphAreaTooBig); message = string.Format(CultureInfo.CurrentCulture, message, relativeArea, relativeAreaMax); throw new OverflowException(message); } #endregion Constructors //------------------------------------------------------ // // Public Methods // //----------------------------------------------------- #region Public Methods /// /// Given a character hit, computes the offset from the leading edge of the glyph run /// to the leading or trailing edge of a caret stop containing the character hit. /// If the glyph run is not hit testable, the distance of 0.0 is returned. /// /// Character hit to compute the distance to. ///The offset from the leading edge of the glyph run /// to the leading or trailing edge of a caret stop containing the character hit. ////// The input character hit is outside of the range specified by the glyph run Unicode string. /// public double GetDistanceFromCaretCharacterHit(CharacterHit characterHit) { CheckInitialized(); // This can only be called on fully initialized GlyphRun IListcaretStops = CaretStops != null && CaretStops.Count != 0 ? CaretStops : new DefaultCaretStopList(CodepointCount); if (characterHit.FirstCharacterIndex < 0 || characterHit.FirstCharacterIndex > CodepointCount) throw new ArgumentOutOfRangeException("characterHit"); int caretStopIndex, codePointsUntilNextStop; FindNearestCaretStop( characterHit.FirstCharacterIndex, caretStops, out caretStopIndex, out codePointsUntilNextStop); // Not a hit testable glyph run. if (caretStopIndex == -1) return 0.0; // Trailing edge of a caret stop that doesn't have a corresponding valid next caret stop. if (codePointsUntilNextStop == -1 && characterHit.TrailingLength != 0) { return 0.0; } // Code point we are measuring the distance to. int caretCodePoint = characterHit.TrailingLength == 0 ? caretStopIndex : caretStopIndex + codePointsUntilNextStop; double distance = 0.0; // Sum up glyph advance widths until the caret code point. IList clusterMap = ClusterMap; if (clusterMap == null) clusterMap = new DefaultClusterMap(CodepointCount); int clusterCodepointStart = 0; int currentCodepoint = clusterCodepointStart; IList advances = AdvanceWidths; for (;;) { ++currentCodepoint; if (currentCodepoint >= clusterMap.Count || clusterMap[currentCodepoint] != clusterMap[clusterCodepointStart]) { // We reached the beginning of the next cluster or the end of the glyph run. // If the codepoint is within the cluster, calculate the partial width and abort the loop. // If the codepoint is past the cluster, accumulate the whole cluster advance width and move on. double clusterWidth = 0; int clusterGlyphEnd; if (currentCodepoint >= clusterMap.Count) clusterGlyphEnd = advances.Count; else clusterGlyphEnd = clusterMap[currentCodepoint]; for (int i = clusterMap[clusterCodepointStart]; i < clusterGlyphEnd; ++i) clusterWidth += advances[i]; if (caretCodePoint < currentCodepoint || currentCodepoint >= clusterMap.Count) { // The caret code point is within a cluster or we are past the end of the run, // sum all glyph advance widths in the cluster // and multiply the result by (caretCodePoint / number of codepoints in the cluster). clusterWidth *= (double)(caretCodePoint - clusterCodepointStart) / (currentCodepoint - clusterCodepointStart); distance += clusterWidth; break; } // The codepoint is past the cluster, accumulate the whole cluster advance width and move on. distance += clusterWidth; clusterCodepointStart = currentCodepoint; } } return distance; } /// /// Given an offset from the leading edge of the glyph run, computes the caret character hit /// that contains the offset. The out bool IsInside parameter describes whether the character hit /// is inside the glyph run. If the hit is outside the glyph run, the character hit represents /// the closest caret character hit within the glyph run. /// /// Distance to compute character hit for. /// isInside is set to true when the character hit /// is inside the glyph run, and to false otherwise. ///The character hit that is closest to the input distance. public CharacterHit GetCaretCharacterHitFromDistance(double distance, out bool isInside) { CheckInitialized(); // This can only be called on fully initialized GlyphRun // Navigate the caret stop array and find a pair of caret stops that contains the distance. IListadvances = AdvanceWidths; IList caretStops = CaretStops != null && CaretStops.Count != 0 ? CaretStops : new DefaultCaretStopList(CodepointCount); IList clusterMap = ClusterMap; if (clusterMap == null) clusterMap = new DefaultClusterMap(CodepointCount); // The following two variables describe the closest caret stop to the left of the input distance. int firstStopIndex = -1; double firstStopAdvance = 0.0; // The following variable describes the closest caret stop to the right of the input distance. int secondStopIndex = -1; // Accumulated advance width just before the current cluster. double currentAdvance = 0.0; // Start index of the cluster we're in. int currentClusterStart = 0; // Since the caretStops array contains clusterMap.Count + 1 elements, // we need to be careful before dereferencing i in the loop body. for (int i = 1; i < caretStops.Count; ++i) { if (i < clusterMap.Count && clusterMap[i] == clusterMap[currentClusterStart]) continue; // We reached the end of an (n:m) cluster. // First, accumulate the overall cluster advance width by summing m glyph advances. ushort lastGlyphInCluster = i < clusterMap.Count ? clusterMap[i] : (ushort)advances.Count; Debug.Assert(clusterMap[currentClusterStart] < lastGlyphInCluster); double clusterAdvance = 0.0; for (int j = clusterMap[currentClusterStart]; j < lastGlyphInCluster; ++j) clusterAdvance += advances[j]; // The overall advance is divided evenly by n code points. clusterAdvance /= i - currentClusterStart; // Go through the individual caret stops and compare them against the input distance for (int j = currentClusterStart; j < i; ++j) { if (caretStops[j]) { if (currentAdvance <= distance) { firstStopIndex = j; firstStopAdvance = currentAdvance; } else { // We found a caret stop to the right of the input distance, // so we're done with enumerating. secondStopIndex = j; goto SecondStopFound; } } currentAdvance += clusterAdvance; } currentClusterStart = i; } // The last iteration is interesting. Because inside the above loop we only look at the caret stops up until i-1, // and there may or may not be a caret stop at the end of a glyph run, // we need to check the last caret stop value and the distance. // The code before SecondStopFound is essentially the reduced version of the loop body above when i == caretStops.Count. // We could modify the loop, but this would result in additional special cases. if (caretStops[caretStops.Count - 1]) { if (currentAdvance > distance) secondStopIndex = caretStops.Count - 1; } SecondStopFound: // First stop is described by firstStopIndex, firstStopAdvance. // Second stop is described by secondStopIndex, currentAdvance. // If both indices are equal to -1, then all caret stop entries except the very last one are set to false. // If the last one is also set to false, the glyph run is not hit testable at all. // If the last one is set to true, we return CharacterHit corresponding to that last caret stop. if (firstStopIndex == -1 && secondStopIndex == -1) { isInside = false; if (caretStops[caretStops.Count - 1]) return new CharacterHit(caretStops.Count - 1, 0); else return new CharacterHit(0, 0); } // Check for case when the first stop is not valid. // This happens when the hit is to the left of the first caret stop. if (firstStopIndex == -1) { isInside = false; // Leading edge of the second stop. return new CharacterHit(secondStopIndex, 0); } // Check for case when the second stop is not valid. // This happens when the hit is to the right of the last caret stop. if (secondStopIndex == -1) { isInside = false; // Trailing edge of the first stop. return new CharacterHit(firstStopIndex, caretStops.Count - 1 - firstStopIndex); } isInside = true; if (distance <= (firstStopAdvance + currentAdvance) / 2.0) { // Leading edge of the first stop. return new CharacterHit(firstStopIndex, 0); } else { // Trailing edge of the first stop. return new CharacterHit(firstStopIndex, secondStopIndex - firstStopIndex); } } /// /// Computes the next valid caret character hit in the logical direction. /// If no further navigation is possible, the returned hit result is the same as input value. /// /// The character hit to compute next hit value for. ///The next valid caret character hit in the logical direction, or /// the input value if no further navigation is possible. public CharacterHit GetNextCaretCharacterHit(CharacterHit characterHit) { CheckInitialized(); // This can only be called on fully initialized GlyphRun IListcaretStops = CaretStops != null && CaretStops.Count != 0 ? CaretStops : new DefaultCaretStopList(CodepointCount); if (characterHit.FirstCharacterIndex < 0 || characterHit.FirstCharacterIndex > CodepointCount) throw new ArgumentOutOfRangeException("characterHit"); int caretStopIndex, codePointsUntilNextStop; FindNearestCaretStop( characterHit.FirstCharacterIndex, caretStops, out caretStopIndex, out codePointsUntilNextStop); // Not a hit testable run, or no next caret code point. if (caretStopIndex == -1 || codePointsUntilNextStop == -1) return characterHit; // If we are at the leading edge, move to the trailing edge of the same code point. if (characterHit.TrailingLength == 0) return new CharacterHit(caretStopIndex, codePointsUntilNextStop); // If the next caret stop is within the glyph run, // move to the trailing edge of it. int nextCaretStopIndex, nextCodePointsUntilNextStop; FindNearestCaretStop( caretStopIndex + codePointsUntilNextStop, caretStops, out nextCaretStopIndex, out nextCodePointsUntilNextStop); // See if the next caret stop is within the glyph run. // If not, no navigation is possible. if (nextCodePointsUntilNextStop == -1) return characterHit; return new CharacterHit(nextCaretStopIndex, nextCodePointsUntilNextStop); } /// /// Computes the previous valid caret character hit in the logical direction. /// If no further navigation is possible, the returned hit result is the same as input value. /// /// The character hit to compute previous hit value for. ///The previous valid caret character hit in the logical direction, or /// the input value if no further navigation is possible. public CharacterHit GetPreviousCaretCharacterHit(CharacterHit characterHit) { CheckInitialized(); // This can only be called on fully initialized GlyphRun IListcaretStops = CaretStops != null && CaretStops.Count != 0 ? CaretStops : new DefaultCaretStopList(CodepointCount); if (characterHit.FirstCharacterIndex < 0 || characterHit.FirstCharacterIndex > CodepointCount) throw new ArgumentOutOfRangeException("characterHit"); int caretStopIndex, codePointsUntilNextStop; FindNearestCaretStop( characterHit.FirstCharacterIndex, caretStops, out caretStopIndex, out codePointsUntilNextStop); if (caretStopIndex == -1) return characterHit; // If we are at the trailing edge, move to the leading edge of the same code point. if (characterHit.TrailingLength != 0) return new CharacterHit(caretStopIndex, 0); // Find the previous caret stop. int previousCaretStopIndex; FindNearestCaretStop( caretStopIndex - 1, caretStops, out previousCaretStopIndex, out codePointsUntilNextStop); // No previous hit, return the original one. if (previousCaretStopIndex == -1 || previousCaretStopIndex == caretStopIndex) return characterHit; return new CharacterHit(previousCaretStopIndex, 0); } #endregion Public Methods //------------------------------------------------------ // // Public Properties // //------------------------------------------------------ #region Public Properties /// /// Advance width from origin of first glyph to far alignment edge of last glyph. /// private double AdvanceWidth { get { double advance = 0; if (_advanceWidths != null) { foreach(double glyphAdvance in _advanceWidths) advance += glyphAdvance; } return advance; } } ////// Distance from the GlyphRun origin to the top of the alignment box. /// private double Ascent { get { // for sideways text, origin is in the middle of the character cell if (IsSideways) return _renderingEmSize * _glyphTypeface.Height / 2.0; return _glyphTypeface.Baseline * _renderingEmSize; } } ////// Distance from top to bottom of alignment box. /// private double Height { get { return _glyphTypeface.Height * _renderingEmSize; } } ////// The baseline origin of the glyph run /// public Point BaselineOrigin { get { CheckInitialized(); return _baselineOrigin; } set { CheckInitializing(); // This can only be set during initialization. _baselineOrigin = value; } } ////// Em size used for rendering. /// public double FontRenderingEmSize { get { CheckInitialized(); return _renderingEmSize; } set { CheckInitializing(); // This can only be set during initialization. _renderingEmSize = value; } } ////// Returns GlyphTypeface for this object. /// public GlyphTypeface GlyphTypeface { get { CheckInitialized(); return _glyphTypeface; } set { CheckInitializing(); // This can only be set during initialization. if (value == null) { throw new ArgumentNullException("value"); } _glyphTypeface = value; } } ////// Determines LTR/RTL reading order and bidi nesting. /// ///The value of bidirectional nesting level. public int BidiLevel { get { CheckInitialized(); return _bidiLevel; } set { CheckInitializing(); // This can only be set during initialization. _bidiLevel = value; } } ////// Returns whether the glyph run is left to right or right to left. /// ///true for LTR, false for RTL. private bool IsLeftToRight { get { return (_bidiLevel & 1) == 0; } } ////// Specifies whether to rotate characters/glyphs 90 degrees anti-clockwise /// and use vertical baseline positioning metrics. /// ///true if the rotation should be applied, false otherwise. public bool IsSideways { get { CheckInitialized(); return (_flags & GlyphRunFlags.IsSideways) != 0; } set { CheckInitializing(); // This can only be set during initialization. if (value) { _flags |= GlyphRunFlags.IsSideways; } else { _flags &= (~GlyphRunFlags.IsSideways); } } } ////// Returns caret stops list for this GlyphRun or null if there is a caret stop for every UTF16 codepoint. /// [CLSCompliant(false)] [TypeConverter(typeof(BoolIListConverter))] public IListCaretStops { get { CheckInitialized(); return _caretStops; } set { CheckInitializing(); // This can only be set during initialization. // The list can be null, empty or non-empty list. // The consistency with other lists would be checked at EndInit() time. _caretStops = value; } } /// /// Returns whether there are any valid caret character hits within the glyph run. /// public bool IsHitTestable { get { CheckInitialized(); // This can only be called on fully initialized GlyphRun if (CaretStops == null || CaretStops.Count == 0) { // When CaretStops property is omitted, there is a caret stop for every UTF16 code point. return true; } foreach (bool caretStop in CaretStops) { if (caretStop) return true; } return false; } } ////// The list that maps characters in the glyph run to glyph indices. /// There is one entry per character in Characters list. /// Each value gives the offset of the first glyph in GlyphIndices /// that represents the corresponding character in Characters. /// Where multiple characters map to a single glyph, or to a glyph group /// that cannot be broken down to map exactly to individual characters, /// the entries for all the characters have the same value: /// the offset of the first glyph that represents this group of characters. /// If the list is null or empty, sequential 1 to 1 mapping is assumed. /// [CLSCompliant(false)] [TypeConverter(typeof(UShortIListConverter))] public IListClusterMap { get { CheckInitialized(); return _clusterMap; } set { CheckInitializing(); // This can only be set during initialization. // The list can be null, empty or non-empty list. // The consistency with other lists would be checked at EndInit() time. _clusterMap = value; } } /// /// Returns the list of UTF16 code points that represent the Unicode content of the glyph run. /// [CLSCompliant(false)] [TypeConverter(typeof(CharIListConverter))] public IListCharacters { get { CheckInitialized(); return _characters; } set { CheckInitializing(); // This can only be set during initialization. // The list can be null, empty or non-empty list. // The consistency with other lists would be checked at EndInit() time. _characters = value; } } /// /// Array of 16 bit glyph numbers that represent this run. /// [CLSCompliant(false)] [TypeConverter(typeof(UShortIListConverter))] public IListGlyphIndices { get { CheckInitialized(); return _glyphIndices; } set { CheckInitializing(); // This can only be set during initialization. // The list must be non-empty list. // The consistency with other lists would be checked at EndInit() time. if (value == null) throw new ArgumentNullException("value"); if (value.Count <= 0) throw new ArgumentException(SR.Get(SRID.CollectionNumberOfElementsMustBeGreaterThanZero), "GlyphIndices"); _glyphIndices = value; } } /// /// The list of advance widths, one for each glyph in GlyphIndices. /// The nominal origin of the nth glyph in the run (n>0) is the nominal origin /// of the n-1th glyph plus the n-1th advance width added along the runs advance vector. /// Base glyphs generally have a non-zero advance width, combining glyphs generally have a zero advance width. /// [CLSCompliant(false)] [TypeConverter(typeof(DoubleIListConverter))] public IListAdvanceWidths { get { CheckInitialized(); return _advanceWidths; } set { CheckInitializing(); // This can only be set during initialization. // The list must be non-empty list. // The consistency with other lists would be checked at EndInit() time. if (value == null) throw new ArgumentNullException("value"); if (value.Count <= 0) throw new ArgumentException(SR.Get(SRID.CollectionNumberOfElementsMustBeGreaterThanZero), "AdvanceWidths"); _advanceWidths = value; } } /// /// Array of glyph offsets. Added to the nominal glyph origin calculated above to generate the final origin for the glyph. /// Base glyphs generally have a glyph offset of (0,0), combining glyphs generally have an offset /// that places them correctly on top of the nearest preceeding base glyph. /// [CLSCompliant(false)] [TypeConverter(typeof(PointIListConverter))] public IListGlyphOffsets { get { CheckInitialized(); return _glyphOffsets; } set { CheckInitializing(); // This can only be set during initialization. // The list can be null, empty or non-empty list. // The consistency with other lists would be checked at EndInit() time. _glyphOffsets = value; } } /// /// Returns the language associated with the glyph run. /// public XmlLanguage Language { get { CheckInitialized(); return _language; } set { CheckInitializing(); // This can only be set during initialization. _language = value; } } ////// Identifies a specific device font for which the GlyphRun has been optimized. When a GlyphRun is /// being rendered on a device that has built-in support for this named font, then the GlyphRun should be rendered using a /// possibly device specific mechanism for selecting that font, and by sending the Unicode codepoints rather than the /// glyph indices. When rendering onto a device that does not include built-in support for the named font, /// this property should be ignored. /// public string DeviceFontName { get { CheckInitialized(); return _deviceFontName; } set { CheckInitializing(); // This can only be set during initialization. _deviceFontName = value; } } #endregion Public Properties ////// Glyph offsets /// The array is indexed starting with InitialGlyph /// internal Point GetGlyphOffset(int i) { if (_glyphOffsets == null || _glyphOffsets.Count == 0) return new Point(0, 0); return _glyphOffsets[i]; } ////// Returns whether GlyphRun contains anything to be drawn. /// Examples of GlyphRun's that don't contain ink are: /// - GlyphRun containing only spaces (" ") /// - GlyphRun wiithout any glyphs in it /// ///true if GlyphRun contains ink, false if it doesn't. private bool HasInk { get { if ((_flags & GlyphRunFlags.HasInkIsCached) != 0) { // // We've already calculated this property before // -- we just need to retrieve the cached bit. // return (_flags & GlyphRunFlags.HasInk) != 0; } else { // // Check if the glyph run contains anything to be drawn. // Cache the result and mark the property as cached. // bool hasInk = _glyphIndices.Count != 0 && !ComputeInkBoundingBox().IsEmpty; _flags |= (hasInk ? GlyphRunFlags.HasInk : GlyphRunFlags.None) | GlyphRunFlags.HasInkIsCached; return hasInk; } } } internal int GlyphCount { get { return _glyphIndices.Count; } } internal int CodepointCount { get { if (_characters != null && _characters.Count != 0) return _characters.Count; if (_clusterMap != null && _clusterMap.Count != 0) return _clusterMap.Count; return _glyphIndices.Count; } } #region Drawing and measurements ////// Computes ink bounding box for the glyph run. /// The rectangle is relative to the glyph run origin. /// ///The ink bounding box of the glyph run public Rect ComputeInkBoundingBox() { CheckInitialized(); // This can only be called on fully initialized GlyphRun if ((_flags & GlyphRunFlags.CacheInkBounds) != 0) { if (_inkBoundingBox != null) { return (Rect) _inkBoundingBox; } } bool italicSimulation = (_glyphTypeface.StyleSimulations & StyleSimulations.ItalicSimulation) != 0; // Special casing Left to Right layout with no italics allows an implementation that is // 12 times faster than the general case. Other combinations of Left to Right and // sideways layout also presents optimization opportunities that need to be implemented. // Italics is used infrequently, so adding the additional 8 routines necessary to handle italics // in combination with the other 4 routines is not justified. if (IsLeftToRight && !IsSideways && !italicSimulation) { Rect rect = ComputeInkBoundingBoxLtoR(); if ((_flags & GlyphRunFlags.CacheInkBounds) != 0) { _inkBoundingBox = rect; } return rect; } double accAdvance = 0; // We don't use Rect and Rect.Union to accumulate the bounding box // because this function is a hot spot and Rect methods perform extra checks that we don't need. double accLeft = double.PositiveInfinity; double accTop = double.PositiveInfinity; double accRight = double.NegativeInfinity; double accBottom = double.NegativeInfinity; for (int i = 0; i < GlyphCount; ++i) { double aw, ah, lsb, rsb, tsb, bsb, baseline; _glyphTypeface.GetGlyphMetrics( _glyphIndices[i], _renderingEmSize, out aw, out ah, out lsb, out rsb, out tsb, out bsb, out baseline ); Point glyphOffset = GetGlyphOffset(i); double originX; if (IsLeftToRight) { originX = accAdvance + glyphOffset.X; } else { // no languages support sideways and right to left in the same run Debug.Assert(!IsSideways); originX = -accAdvance - (aw + glyphOffset.X); } accAdvance += _advanceWidths[i]; double horBaselineOriginY = -glyphOffset.Y; double left, right, bottom, top; if (IsSideways) { horBaselineOriginY += aw / 2.0; bottom = horBaselineOriginY - lsb; top = horBaselineOriginY - aw + rsb; left = originX + tsb; right = left + ah - tsb - bsb; } else { left = originX + lsb; right = originX + aw - rsb; bottom = horBaselineOriginY + baseline; top = bottom - ah + tsb + bsb; } // extend the ink box for the glyph to account for italic simulation if (italicSimulation) { // the italic simulation is always applied relative to horizontal baseline right += Sin20 * (horBaselineOriginY - top); left -= Sin20 * (bottom - horBaselineOriginY); } // skip blank glyphs, as they don't contain ink if (left + InkMetricsEpsilon >= right || top + InkMetricsEpsilon >= bottom) continue; if (accLeft > left) accLeft = left; if (accTop > top) accTop = top; if (accRight < right) accRight = right; if (accBottom < bottom) accBottom = bottom; } Rect bounds; if (accLeft > accRight) { bounds = Rect.Empty; } else { bounds = new Rect( accLeft, accTop, accRight - accLeft, accBottom - accTop ); } if ((_flags & GlyphRunFlags.CacheInkBounds) != 0) { _inkBoundingBox = bounds; } return bounds; } private Rect ComputeInkBoundingBoxLtoR() { // We don't use Rect and Rect.Union to accumulate the bounding box // because this function is a hot spot and Rect methods perform extra checks that we don't need. double accLeft = double.PositiveInfinity; double accTop = double.PositiveInfinity; double accRight = double.NegativeInfinity; double accBottom = double.NegativeInfinity; double accAdvance = 0; int glyphCount = GlyphCount; if (GlyphOffsets != null) { for (int i = 0; i < glyphCount; ++i) { double aw, ah, lsb, rsb, tsb, bsb, baseline; _glyphTypeface.GetGlyphMetrics( _glyphIndices[i], _renderingEmSize, out aw, out ah, out lsb, out rsb, out tsb, out bsb, out baseline ); Point glyphOffset = GetGlyphOffset(i); double originX = accAdvance + glyphOffset.X; accAdvance += _advanceWidths[i]; double horBaselineOriginY = -glyphOffset.Y; double left, right, bottom, top; left = originX + lsb; right = originX + aw - rsb; bottom = horBaselineOriginY + baseline; top = bottom - ah + tsb + bsb; // skip blank glyphs, as they don't contain ink if (left + InkMetricsEpsilon >= right || top + InkMetricsEpsilon >= bottom) continue; if (accLeft > left) accLeft = left; if (accTop > top) accTop = top; if (accRight < right) accRight = right; if (accBottom < bottom) accBottom = bottom; } } else { for (int i = 0; i < glyphCount; ++i) { double aw, ah, lsb, rsb, tsb, bsb, baseline; _glyphTypeface.GetGlyphMetrics( _glyphIndices[i], _renderingEmSize, out aw, out ah, out lsb, out rsb, out tsb, out bsb, out baseline ); double left, right, top; left = accAdvance + lsb; right = accAdvance + aw - rsb; top = baseline - ah + tsb + bsb; accAdvance += _advanceWidths[i]; // skip blank glyphs, as they don't contain ink if (left + InkMetricsEpsilon >= right || top + InkMetricsEpsilon >= baseline) continue; if (accLeft > left) accLeft = left; if (accTop > top) accTop = top; if (accRight < right) accRight = right; if (accBottom < baseline) accBottom = baseline; } } if (accLeft > accRight) return Rect.Empty; return new Rect( accLeft, accTop, accRight - accLeft, accBottom - accTop ); } ////// Obtains geometry for the glyph run. /// ///The geometry returned contains the combined geometry of all glyphs in the glyph run. /// Overlapping contours are merged by performing a Boolean union operation. public Geometry BuildGeometry() { CheckInitialized(); // This can only be called on fully initialized GlyphRun GeometryGroup accumulatedGeometry = null; double accAdvance = 0; for (int i = 0; i < GlyphCount; ++i) { ushort glyphIndex = _glyphIndices[i]; double originX; if (IsLeftToRight) { originX = accAdvance; originX += GetGlyphOffset(i).X; } else { // no languages support sideways and right to left in the same run Debug.Assert(!IsSideways); double nominalAdvance = _glyphTypeface.GetAdvanceWidth(glyphIndex) * _renderingEmSize; originX = -accAdvance; originX -= (nominalAdvance + GetGlyphOffset(i).X); } accAdvance += _advanceWidths[i]; double originY = -GetGlyphOffset(i).Y; Geometry glyphGeometry = _glyphTypeface.ComputeGlyphOutline(glyphIndex, IsSideways, _renderingEmSize); if (glyphGeometry.IsEmpty()) continue; // transform glyphGeometry to the glyph origin glyphGeometry.Transform = new TranslateTransform(originX + _baselineOrigin.X, originY + _baselineOrigin.Y); if (accumulatedGeometry == null) { accumulatedGeometry = new GeometryGroup(); accumulatedGeometry.FillRule = FillRule.Nonzero; } accumulatedGeometry.Children.Add(glyphGeometry.GetOutlinedPathGeometry(RelativeFlatteningTolerance, ToleranceType.Relative)); } // Make sure to always return Geometry.Empty from public methods for empty geometries. if (accumulatedGeometry == null || accumulatedGeometry.IsEmpty()) return Geometry.Empty; return accumulatedGeometry; } ////// Computes the alignment box for the glyph run. /// The alignment box is relative to origin. /// ///The alignment box for the glyph run. public Rect ComputeAlignmentBox() { CheckInitialized(); // This can only be called on fully initialized GlyphRun if (IsLeftToRight) { return new Rect( 0, -Ascent, AdvanceWidth, Height ); } else { // cache AdvanceWidth value in a local variable because it involves a loop double advanceWidth = AdvanceWidth; return new Rect( -advanceWidth, -Ascent, advanceWidth, Height ); } } ////// Temporary helper to draw a glyph run background. /// We hope to remove all uses of it, as fundamentally this is not the right way /// to handle background drawing. /// internal void EmitBackground(DrawingContext dc, Brush backgroundBrush) { if (backgroundBrush != null) { Rect backgroundRect; if (IsLeftToRight) { backgroundRect = new Rect( _baselineOrigin.X, _baselineOrigin.Y - Ascent, AdvanceWidth, Height ); } else { backgroundRect = new Rect( _baselineOrigin.X - AdvanceWidth, _baselineOrigin.Y - Ascent, AdvanceWidth, Height ); } dc.DrawRectangle( backgroundBrush, null, backgroundRect ); } } #endregion Drawing and measurements #region DUCE.IResource implementation ////// A structure to keep two scaling ratios fetched from given Matrix. /// internal struct Scale { internal Scale(ref Matrix matrix) { double m11 = matrix.M11; double m12 = matrix.M12; double m21 = matrix.M21; double m22 = matrix.M22; // Calculate redundant data. _baseVectorX = Math.Sqrt(m11 * m11 + m12 * m12); // Check for wrong matrix. if (DoubleUtil.IsNaN(_baseVectorX)) _baseVectorX = 0; _baseVectorY = _baseVectorX == 0 ? 0 : Math.Abs(m11 * m22 - m12 * m21) / _baseVectorX; if (DoubleUtil.IsNaN(_baseVectorY)) _baseVectorY = 0; } internal bool IsValid { get { return _baseVectorX != 0 && _baseVectorY != 0; } } internal bool IsSame(ref Scale scale) { // // allow some imprecision that can appear because // of matrix computations. // return _baseVectorX * 0.999999999 <= scale._baseVectorX && _baseVectorX * 1.000000001 >= scale._baseVectorX && _baseVectorY * 0.999999999 <= scale._baseVectorY && _baseVectorY * 1.000000001 >= scale._baseVectorY; } internal double _baseVectorX, _baseVectorY; } private DUCE.MultiChannelResource _mcr = new DUCE.MultiChannelResource(); ////// Generate a series of requests to create or update /// slave glyph run resource and all depending data. /// DUCE.ResourceHandle DUCE.IResource.AddRefOnChannel(DUCE.Channel channel) { CheckInitialized(); // This can only be called on fully initialized GlyphRun using (CompositionEngineLock.Acquire()) { if (_mcr.CreateOrAddRefOnChannel(channel, DUCE.ResourceType.TYPE_GLYPHRUN)) { CreateOnChannel(channel); } return _mcr.GetHandle(channel); } } ////// Generates request to delete slave glyph run resource. /// void DUCE.IResource.ReleaseOnChannel(DUCE.Channel channel) { CheckInitialized(); // This can only be called on fully initialized GlyphRun using (CompositionEngineLock.Acquire()) { _mcr.ReleaseOnChannel(channel); } } ////// This is only implemented by Visual and Visual3D. /// void DUCE.IResource.RemoveChildFromParent(DUCE.IResource parent, DUCE.Channel channel) { throw new NotImplementedException(); } ////// This is only implemented by Visual and Visual3D. /// DUCE.ResourceHandle DUCE.IResource.Get3DHandle(DUCE.Channel channel) { throw new NotImplementedException(); } ////// Returns current resource handle, allocated recently by AddRefOnChannel. /// DUCE.ResourceHandle DUCE.IResource.GetHandle(DUCE.Channel channel) { CheckInitialized(); // This can only be called on fully initialized GlyphRun return _mcr.GetHandle(channel); } int DUCE.IResource.GetChannelCount() { return _mcr.GetChannelCount(); } DUCE.Channel DUCE.IResource.GetChannel(int index) { return _mcr.GetChannel(index); } ////// Send to channel command sequence to create slave resource. /// ////// Critical: This code acceses an unsafe code block /// TreatAsSafe: This does not expose any data and the unsafe code blocks /// needs to be verified for correctness /// [SecurityCritical,SecurityTreatAsSafe] private void CreateOnChannel(DUCE.Channel channel) { Debug.Assert(_glyphTypeface != null); int glyphCount = GlyphCount; string fontFileName = _glyphTypeface.FontSource.GetUriString(); // // The InkBoundingBox + the Origin produce the true InkBoundingBox. // // Not sure why the bounding box code doesn't adjust for this when you // ask for the bounding box, instead everything // that is interested in the bounding box has to do this calculation. // Rect adjustedInkBoundingBox = ComputeInkBoundingBox(); if (!adjustedInkBoundingBox.IsEmpty) { adjustedInkBoundingBox.Offset((Vector)BaselineOrigin); } DUCE.MILCMD_GLYPHRUN_CREATE command; command.Type = MILCMD.MilCmdGlyphRunCreate; command.Handle = _mcr.GetHandle(channel); command.hGlyphCache = channel.GlyphCache.Handle; command.FontFaceIndex = _glyphTypeface.FaceIndex; command.GlyphRunFlags = ComposeFlags(); command.Origin.X = (float)_baselineOrigin.X; command.Origin.Y = (float)_baselineOrigin.Y; command.MuSize = (float)_renderingEmSize; command.ManagedBounds = (Rect)adjustedInkBoundingBox; command.FontFileNameLength = checked((UInt16)(fontFileName.Length + 1)); command.GlyphCount = checked((UInt16)glyphCount); unsafe { // calculate variable data size int varDataSize = (fontFileName.Length + 1) * sizeof(char); // '\0' terminating char if (_glyphOffsets != null && _glyphOffsets.Count != 0) { varDataSize += glyphCount * (2 * sizeof(float)); // positionXY } else { varDataSize += (glyphCount - 1) * sizeof(float); // positionX } varDataSize += glyphCount * sizeof(ushort); // glyph indices channel.BeginCommand( (byte*)&command, sizeof(DUCE.MILCMD_GLYPHRUN_CREATE), varDataSize ); { // Copy the font filename string char* pFontFileName = stackalloc char[fontFileName.Length + 1]; for (int i = 0; i < fontFileName.Length; i++) { pFontFileName[i] = fontFileName[i]; } pFontFileName[fontFileName.Length] = '\0'; channel.AppendCommandData((byte*)pFontFileName, (fontFileName.Length + 1) * sizeof(char)); // fill variable data block double rcmuSize = 1.0 / _renderingEmSize; if (Double.IsInfinity(rcmuSize)) { // Protect against extremely small _renderingEmSize: // denormalized numbers like 5E-324 can cause overflow on // calculating reciprocal value. rcmuSize = 0; } double xRatio = ((command.GlyphRunFlags & (uint)MilGlyphRun.IsLeftToRight) != 0) ? rcmuSize : -rcmuSize; if (_glyphOffsets != null && _glyphOffsets.Count != 0) { double yRatio = -rcmuSize; double accAdvances = 0; if (glyphCount <= MaxStackAlloc / (2 * sizeof(float))) { // glyph count small enough, send all data at once float* pGlyphPositions = stackalloc float[glyphCount * 2]; for (int i = 0; i < glyphCount; i++) { double x = _glyphOffsets[i].X + accAdvances; double y = _glyphOffsets[i].Y; accAdvances += _advanceWidths[i]; pGlyphPositions[2 * i] = (float)(x * xRatio); pGlyphPositions[2 * i + 1] = (float)(y * yRatio); } channel.AppendCommandData((byte*)pGlyphPositions, glyphCount * (2 * sizeof(float))); } else { // glyph count is not small, use per-glyph transmitting float* pGlyphPositions = stackalloc float[2]; for (int i = 0; i < glyphCount; i++) { double x = _glyphOffsets[i].X + accAdvances; double y = _glyphOffsets[i].Y; accAdvances += _advanceWidths[i]; pGlyphPositions[0] = (float)(x * xRatio); pGlyphPositions[1] = (float)(y * yRatio); channel.AppendCommandData((byte*)pGlyphPositions, 2 * sizeof(float)); } } } else if (glyphCount > 1) { // accumulate advance widths and convert them to "glyph space" double accAdvances = 0; if (glyphCount <= MaxStackAlloc / sizeof(float)) { // glyph count small enough, send all data at once float* pPositionX = stackalloc float[glyphCount - 1]; // skipping first glyph position that's always zero for (int i = 0; i < (glyphCount - 1); ++i) { accAdvances += _advanceWidths[i]; pPositionX[i] = (float)(accAdvances * xRatio); } channel.AppendCommandData((byte*)pPositionX, (glyphCount - 1) * sizeof(float)); } else { // glyph count is not small, use per-glyph transmitting for (int i = 0; i < (glyphCount - 1); ++i) { accAdvances += _advanceWidths[i]; float positionX = (float)(accAdvances * xRatio); channel.AppendCommandData((byte*)&positionX, sizeof(float)); } } } { // transmit glyph indices if (glyphCount <= MaxStackAlloc / sizeof(ushort)) { // glyph count small enough, send all data at once ushort* pGlyphIndices = stackalloc ushort[glyphCount]; for (int i = 0; i < glyphCount; ++i) { pGlyphIndices[i] = _glyphIndices[i]; } channel.AppendCommandData((byte*)pGlyphIndices, glyphCount * sizeof(ushort)); } else { // glyph count is not small, use per-glyph transmitting for (int i = 0; i < glyphCount; ++i) { ushort glyphIndex = _glyphIndices[i]; channel.AppendCommandData((byte*)&glyphIndex, sizeof(ushort)); } } } } channel.EndCommand(); } } ////// Gather flags that affect: /// - glyph run rendering /// - glyph rasterization /// - the way how glyph run data is packed /// private UInt16 ComposeFlags() { UInt16 flags = 0; if (_glyphTypeface.FontTechnology != FontTechnology.PostscriptOpenType) { flags |= (UInt16)MilGlyphRun.IsTrueType; FontFaceLayoutInfo.RenderingHints renderingHints = _glyphTypeface.RenderingHints; // When dealing with East Asian fonts containing embedded bitmaps // we apply a special 6x5 mode with hinting at em square and enhanced constract algorithms. if (renderingHints == FontFaceLayoutInfo.RenderingHints.LegacyEastAsian) { flags |= (UInt16)MilGlyphRun.ForceVAA | (UInt16)MilGlyphRun.VerticalDropOut | (UInt16)MilGlyphRun.OverContrast; } else { flags |= (UInt16)MilGlyphRun.Hinting; } } else { // For CFF fonts we always use 6x5 overscale hinted mode, because stem thickening happens in the CFF rasterizer. flags |= (UInt16)MilGlyphRun.Hinting | (UInt16)MilGlyphRun.ForceVAA; } StyleSimulations styleSimulations = _glyphTypeface.StyleSimulations; if ((styleSimulations & StyleSimulations.BoldSimulation) != 0) flags |= (UInt16)MilGlyphRun.BoldSimulation; if ((styleSimulations & StyleSimulations.ItalicSimulation) != 0) flags |= (UInt16)MilGlyphRun.ItalicSimulation; if (IsSideways) flags |= (UInt16)MilGlyphRun.Sideways; // if ((_bidiLevel & 1) == 0) flags |= (UInt16)MilGlyphRun.IsLeftToRight; if (_glyphOffsets != null && _glyphOffsets.Count != 0) flags |= (UInt16)MilGlyphRun.HasYPositions; // Encode font contrast adjustment into 3 bits masked by MilGlyphRunPrecontrastLevel. UInt16 fontContrastAdjustment = (UInt16)(_glyphTypeface.FontContrastAdjustment + (int)MilGlyphRun.PrecontrastOffset); // If the resulting value spills outside the mask, this indicates a code bug in the GlyphTypeface code. Debug.Assert(((fontContrastAdjustment << (int)MilGlyphRun.PrecontrastShift) & ~(UInt16)MilGlyphRun.PrecontrastLevel) == 0); flags |= (UInt16)(fontContrastAdjustment << (int)MilGlyphRun.PrecontrastShift); return flags; } #endregion DUCE.IResource implementation #region Hit testing ////// Given a code point index in the caret stop array, finds the nearest pair of caret stops. /// /// Character index to start the search from. Doesn't have to be snapped. /// GlyphRun CaretStops array. Guaranteed to be non-null. /// Nearest caret stop index, or -1 if there are no caret stops. /// Code points until the next caret stop, or -1 if there is no next caret stop. private void FindNearestCaretStop( int characterIndex, IListcaretStops, out int caretStopIndex, out int codePointsUntilNextStop) { caretStopIndex = -1; codePointsUntilNextStop = -1; if (characterIndex < 0 || characterIndex >= caretStops.Count) return; // Find the closest caret stop at the character index or to the left of it. for (int i = characterIndex; i >= 0; --i) { if (caretStops[i]) { caretStopIndex = i; break; } } // Couldn't find a caret stop at the character index or to the left of it. // Search to the right. if (caretStopIndex == -1) { for (int i = characterIndex + 1; i < caretStops.Count; ++i) { if (caretStops[i]) { caretStopIndex = i; break; } } } // No caret stops found, the glyph run is not hit testable. if (caretStopIndex == -1) { return; } for (int lastStop = caretStopIndex + 1; lastStop < caretStops.Count; ++lastStop) { if (caretStops[lastStop]) { // There is a next caret stop. codePointsUntilNextStop = lastStop - caretStopIndex; return; } } // There is no next caret stop. } /// /// This class implements behavior of a Boolean list that contains all true values. /// This allows us to have a single code path in hit testing API. /// private class DefaultCaretStopList : IList{ public DefaultCaretStopList(int codePointCount) { _count = codePointCount + 1; } #region IList Members public int IndexOf(bool item) { throw new NotSupportedException(); } public void Insert(int index, bool item) { throw new NotSupportedException(); } public bool this[int index] { get { return true; } set { throw new NotSupportedException(); } } public void RemoveAt(int index) { throw new NotSupportedException(); } #endregion #region ICollection Members public void Add(bool item) { throw new NotSupportedException(); } public void Clear() { throw new NotSupportedException(); } public bool Contains(bool item) { throw new NotSupportedException(); } public void CopyTo(bool[] array, int arrayIndex) { throw new NotSupportedException(); } public int Count { get { return _count; } } public bool IsReadOnly { get { return true; } } public bool Remove(bool item) { throw new NotSupportedException(); } #endregion #region IEnumerable Members IEnumerator IEnumerable .GetEnumerator() { throw new NotSupportedException(); } #endregion #region IEnumerable Members IEnumerator IEnumerable.GetEnumerator() { throw new NotSupportedException(); } #endregion private int _count; } /// /// This class implements behavior of a 1:1 cluster map. /// This allows us to have a single code path in hit testing API. /// private class DefaultClusterMap : IList{ public DefaultClusterMap(int count) { _count = count; } #region IList Members public int IndexOf(ushort item) { throw new NotSupportedException(); } public void Insert(int index, ushort item) { throw new NotSupportedException(); } public ushort this[int index] { get { return (ushort)index; } set { throw new NotSupportedException(); } } public void RemoveAt(int index) { throw new NotSupportedException(); } #endregion #region ICollection Members public void Add(ushort item) { throw new NotSupportedException(); } public void Clear() { throw new NotSupportedException(); } public bool Contains(ushort item) { throw new NotSupportedException(); } public void CopyTo(ushort[] array, int arrayIndex) { throw new NotSupportedException(); } public int Count { get { return _count; } } public bool IsReadOnly { get { return true; } } public bool Remove(ushort item) { throw new NotSupportedException(); } #endregion #region IEnumerable Members IEnumerator IEnumerable .GetEnumerator() { throw new NotSupportedException(); } #endregion #region IEnumerable Members IEnumerator IEnumerable.GetEnumerator() { throw new NotSupportedException(); } #endregion private int _count; } #endregion Hit testing #region ISupportInitialize interface for Xaml serialization void ISupportInitialize.BeginInit() { if (IsInitialized) { // Cannot initialize a GlyphRun that is completely initialized. throw new InvalidOperationException(SR.Get(SRID.OnlyOneInitialization)); } if (IsInitializing) { // Cannot initialize a GlyphRun that is already being initialized. throw new InvalidOperationException(SR.Get(SRID.InInitialization)); } IsInitializing = true; } void ISupportInitialize.EndInit() { if (!IsInitializing) { // Cannot EndInit a GlyphRun that is not being initialized. throw new InvalidOperationException(SR.Get(SRID.NotInInitialization)); } // // Fully initilize the GlyphRun. The method will check for consistency // between all the properties. // Initialize( _glyphTypeface, _bidiLevel, (_flags & GlyphRunFlags.IsSideways) != 0, _renderingEmSize, _glyphIndices, _baselineOrigin, (_advanceWidths == null ? null : new ThousandthOfEmRealDoubles(_renderingEmSize, _advanceWidths)), (_glyphOffsets == null ? null : new ThousandthOfEmRealPoints(_renderingEmSize, _glyphOffsets)), _characters, _deviceFontName, _clusterMap, _caretStops, _language, true // throwIfOverflow ); // User should be able to fix errors that are only caught at EndInit() time. So set Initializing flag to // false after Initialization succeeds. IsInitializing = false; } private void CheckInitialized() { if (!IsInitialized) { throw new InvalidOperationException(SR.Get(SRID.InitializationIncomplete)); } // Ensure the bits are set consistently. The object cannot be in both states. Debug.Assert(!IsInitializing); } private void CheckInitializing() { if (!IsInitializing) { throw new InvalidOperationException(SR.Get(SRID.NotInInitialization)); } // Ensure the bits are set consistently. The object cannot be in both states. Debug.Assert(!IsInitialized); } private bool IsInitializing { get { return (_flags & GlyphRunFlags.IsInitializing) != 0; } set { if (value) { _flags |= GlyphRunFlags.IsInitializing; } else { _flags &= (~GlyphRunFlags.IsInitializing); } } } private bool IsInitialized { get { return (_flags & GlyphRunFlags.IsInitialized) != 0; } set { if (value) { _flags |= GlyphRunFlags.IsInitialized; } else { _flags &= (~GlyphRunFlags.IsInitialized); } } } #endregion //----------------------------------------------------- // // Private Enumerations // //------------------------------------------------------ #region Private Enumerations /// /// Glyph run flags. /// [Flags] private enum GlyphRunFlags : byte { ////// No flags set. /// It also represents the state in which the GlyphRun has not been initialized. /// At this state, all operations on the object would cause InvalidOperationException. /// The object can only transit to 'IsInitializing' state with BeginInit() call. /// None = 0x00, ////// Set to display the GlyphRun sideways. /// IsSideways = 0x01, ////// Set if the glyph run contains anything to be drawn. /// HasInk = 0x02, ////// Set if the HasInk bit above has already been calculated and cached. /// HasInkIsCached = 0x04, ////// The state in which the GlyphRun object is fully initialized. At this state the object /// is fully functional. There is no valid transition out of the state. /// IsInitialized = 0x08, ////// The state in which the GlyphRun is being initialized. At this state, user can /// set values into the required properties. The object can only transit to 'IsInitialized' state /// with EndInit() call. /// IsInitializing = 0x10, ////// Caching ink bounds /// CacheInkBounds = 0x20, } #endregion Private Enumerations //----------------------------------------------------- // // Private Fields // //----------------------------------------------------- #region Private Fields private Point _baselineOrigin; private GlyphRunFlags _flags; private double _renderingEmSize; private IList_glyphIndices; private IList _advanceWidths; private IList _glyphOffsets; private int _bidiLevel; private GlyphTypeface _glyphTypeface; private IList _characters; private IList _clusterMap; private IList _caretStops; private XmlLanguage _language; private string _deviceFontName; private object _inkBoundingBox; // Used when CacheInkBounds is on // the sine of 20 degrees private const double Sin20 = 0.34202014332566873304409961468226; // This is the precision that is used to decide that glyph metrics are equal, // for example when detecting blank glyphs. // The chosen value is greater than typical floating point precision loss // but smaller than typical design font unit (1/1024th or 1/2048th). private const double InkMetricsEpsilon = 0.0000001; // Dummy font hinting size private const double DefaultFontHintingSize = 12.0; // Tolerance for flattening Bezier curves when calling GetOutlinedPathGeometry. internal static double RelativeFlatteningTolerance = 0.01; // The constants that delimit glyph run size. // Should correspond to unmanaged ones in GlyphRunCore.h. internal const int MaxGlyphCount = 0xFFFF; internal const int OverscaledCoordMax = 0xFFFFF; internal const int GeometryThreshold = 100; internal const int MaxOvescale = 8; internal const int MaxOverscaledBitsInGlyphRun = 800000000; // 800 MBits = 100 MBytes internal const int MaxStackAlloc = 1024; #endregion Private Fields } } // File provided for Reference Use Only by Microsoft Corporation (c) 2007.
Link Menu
This book is available now!
Buy at Amazon US or
Buy at Amazon UK
- GenericPrincipal.cs
- VirtualStackFrame.cs
- PersonalizationState.cs
- SharedConnectionWorkflowTransactionService.cs
- Activity.cs
- DecoratedNameAttribute.cs
- ModulesEntry.cs
- PropertyGridEditorPart.cs
- TraceHandler.cs
- XmlException.cs
- ToolboxBitmapAttribute.cs
- TraceUtility.cs
- List.cs
- DesignerObjectListAdapter.cs
- AcceleratedTokenProvider.cs
- Menu.cs
- CaretElement.cs
- ClientRuntimeConfig.cs
- SelectionUIService.cs
- DataServiceRequestException.cs
- SchemaImporterExtension.cs
- DrawingBrush.cs
- RepeatInfo.cs
- DataServiceHost.cs
- SQLMoney.cs
- TextEditorParagraphs.cs
- ScrollData.cs
- XamlTypeMapper.cs
- ConfigXmlComment.cs
- TypeExtensionSerializer.cs
- X509SecurityToken.cs
- CryptoApi.cs
- ActionItem.cs
- TdsParserHelperClasses.cs
- PeerApplication.cs
- CharKeyFrameCollection.cs
- KoreanCalendar.cs
- RealizedColumnsBlock.cs
- Int16Converter.cs
- SpoolingTask.cs
- DigitShape.cs
- XmlILConstructAnalyzer.cs
- DefaultParameterValueAttribute.cs
- Margins.cs
- FormViewRow.cs
- ContainerParaClient.cs
- BroadcastEventHelper.cs
- StaticExtensionConverter.cs
- RuntimeVariablesExpression.cs
- EmptyReadOnlyDictionaryInternal.cs
- HuffmanTree.cs
- HtmlInputImage.cs
- CngUIPolicy.cs
- CompositionAdorner.cs
- ExpressionBuilder.cs
- TransformerInfo.cs
- ResourcePool.cs
- Attachment.cs
- MulticastOption.cs
- ValidationSummaryDesigner.cs
- DifferencingCollection.cs
- XmlElement.cs
- InternalConfigHost.cs
- HelpProvider.cs
- __ConsoleStream.cs
- MultiPropertyDescriptorGridEntry.cs
- RemoteX509Token.cs
- FileClassifier.cs
- SByteConverter.cs
- ConstraintStruct.cs
- AnchoredBlock.cs
- PtsPage.cs
- DbConnectionPool.cs
- bidPrivateBase.cs
- MetabaseSettingsIis7.cs
- DataGridViewControlCollection.cs
- CompilationUtil.cs
- ConnectorDragDropGlyph.cs
- DownloadProgressEventArgs.cs
- ImageAutomationPeer.cs
- WebPartMinimizeVerb.cs
- webeventbuffer.cs
- HttpPostClientProtocol.cs
- ReadOnlyCollection.cs
- _SslState.cs
- CodeAttributeDeclaration.cs
- StorageInfo.cs
- StyleSelector.cs
- SQLDecimalStorage.cs
- FocusChangedEventArgs.cs
- _CacheStreams.cs
- InstancePersistence.cs
- ServiceHostingEnvironment.cs
- _ChunkParse.cs
- _UriSyntax.cs
- UserNamePasswordServiceCredential.cs
- MatrixKeyFrameCollection.cs
- SecurityHelper.cs
- TripleDES.cs
- DBConnectionString.cs