You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
 
 
 
 

7509 lines
322 KiB

#define TMP_PRESENT
using UnityEngine;
using UnityEngine.UI;
using UnityEngine.Events;
using UnityEngine.EventSystems;
using System;
using System.Text;
using System.Collections;
using System.Collections.Generic;
namespace TMPro
{
public interface ITextElement
{
Material sharedMaterial { get; }
void Rebuild(CanvasUpdate update);
int GetInstanceID();
}
public enum TextAlignmentOptions
{
TopLeft = _HorizontalAlignmentOptions.Left | _VerticalAlignmentOptions.Top,
Top = _HorizontalAlignmentOptions.Center | _VerticalAlignmentOptions.Top,
TopRight = _HorizontalAlignmentOptions.Right | _VerticalAlignmentOptions.Top,
TopJustified = _HorizontalAlignmentOptions.Justified | _VerticalAlignmentOptions.Top,
TopFlush = _HorizontalAlignmentOptions.Flush | _VerticalAlignmentOptions.Top,
TopGeoAligned = _HorizontalAlignmentOptions.Geometry | _VerticalAlignmentOptions.Top,
Left = _HorizontalAlignmentOptions.Left | _VerticalAlignmentOptions.Middle,
Center = _HorizontalAlignmentOptions.Center | _VerticalAlignmentOptions.Middle,
Right = _HorizontalAlignmentOptions.Right | _VerticalAlignmentOptions.Middle,
Justified = _HorizontalAlignmentOptions.Justified | _VerticalAlignmentOptions.Middle,
Flush = _HorizontalAlignmentOptions.Flush | _VerticalAlignmentOptions.Middle,
CenterGeoAligned = _HorizontalAlignmentOptions.Geometry | _VerticalAlignmentOptions.Middle,
BottomLeft = _HorizontalAlignmentOptions.Left | _VerticalAlignmentOptions.Bottom,
Bottom = _HorizontalAlignmentOptions.Center | _VerticalAlignmentOptions.Bottom,
BottomRight = _HorizontalAlignmentOptions.Right | _VerticalAlignmentOptions.Bottom,
BottomJustified = _HorizontalAlignmentOptions.Justified | _VerticalAlignmentOptions.Bottom,
BottomFlush = _HorizontalAlignmentOptions.Flush | _VerticalAlignmentOptions.Bottom,
BottomGeoAligned = _HorizontalAlignmentOptions.Geometry | _VerticalAlignmentOptions.Bottom,
BaselineLeft = _HorizontalAlignmentOptions.Left | _VerticalAlignmentOptions.Baseline,
Baseline = _HorizontalAlignmentOptions.Center | _VerticalAlignmentOptions.Baseline,
BaselineRight = _HorizontalAlignmentOptions.Right | _VerticalAlignmentOptions.Baseline,
BaselineJustified = _HorizontalAlignmentOptions.Justified | _VerticalAlignmentOptions.Baseline,
BaselineFlush = _HorizontalAlignmentOptions.Flush | _VerticalAlignmentOptions.Baseline,
BaselineGeoAligned = _HorizontalAlignmentOptions.Geometry | _VerticalAlignmentOptions.Baseline,
MidlineLeft = _HorizontalAlignmentOptions.Left | _VerticalAlignmentOptions.Geometry,
Midline = _HorizontalAlignmentOptions.Center | _VerticalAlignmentOptions.Geometry,
MidlineRight = _HorizontalAlignmentOptions.Right | _VerticalAlignmentOptions.Geometry,
MidlineJustified = _HorizontalAlignmentOptions.Justified | _VerticalAlignmentOptions.Geometry,
MidlineFlush = _HorizontalAlignmentOptions.Flush | _VerticalAlignmentOptions.Geometry,
MidlineGeoAligned = _HorizontalAlignmentOptions.Geometry | _VerticalAlignmentOptions.Geometry,
CaplineLeft = _HorizontalAlignmentOptions.Left | _VerticalAlignmentOptions.Capline,
Capline = _HorizontalAlignmentOptions.Center | _VerticalAlignmentOptions.Capline,
CaplineRight = _HorizontalAlignmentOptions.Right | _VerticalAlignmentOptions.Capline,
CaplineJustified = _HorizontalAlignmentOptions.Justified | _VerticalAlignmentOptions.Capline,
CaplineFlush = _HorizontalAlignmentOptions.Flush | _VerticalAlignmentOptions.Capline,
CaplineGeoAligned = _HorizontalAlignmentOptions.Geometry | _VerticalAlignmentOptions.Capline
};
/// <summary>
/// Internal horizontal text alignment options.
/// </summary>
public enum _HorizontalAlignmentOptions
{
Left = 0x1, Center = 0x2, Right = 0x4, Justified = 0x8, Flush = 0x10, Geometry = 0x20
}
/// <summary>
/// Internal vertical text alignment options.
/// </summary>
public enum _VerticalAlignmentOptions
{
Top = 0x100, Middle = 0x200, Bottom = 0x400, Baseline = 0x800, Geometry = 0x1000, Capline = 0x2000,
}
/// <summary>
/// Flags controlling what vertex data gets pushed to the mesh.
/// </summary>
public enum TextRenderFlags
{
DontRender = 0x0,
Render = 0xFF
};
public enum TMP_TextElementType { Character, Sprite };
public enum MaskingTypes { MaskOff = 0, MaskHard = 1, MaskSoft = 2 }; //, MaskTex = 4 };
public enum TextOverflowModes { Overflow = 0, Ellipsis = 1, Masking = 2, Truncate = 3, ScrollRect = 4, Page = 5, Linked = 6 };
public enum MaskingOffsetMode { Percentage = 0, Pixel = 1 };
public enum TextureMappingOptions { Character = 0, Line = 1, Paragraph = 2, MatchAspect = 3 };
public enum FontStyles { Normal = 0x0, Bold = 0x1, Italic = 0x2, Underline = 0x4, LowerCase = 0x8, UpperCase = 0x10, SmallCaps = 0x20, Strikethrough = 0x40, Superscript = 0x80, Subscript = 0x100, Highlight = 0x200 };
public enum FontWeights { Thin = 100, ExtraLight = 200, Light = 300, Normal = 400, Medium = 500, SemiBold = 600, Bold = 700, Heavy = 800, Black = 900 };
public enum TagUnits { Pixels = 0, FontUnits = 1, Percentage = 2 };
public enum TagType { None = 0x0, NumericalValue = 0x1, StringValue = 0x2, ColorValue = 0x4 };
/// <summary>
/// Base class which contains common properties and functions shared between the TextMeshPro and TextMeshProUGUI component.
/// </summary>
public abstract class TMP_Text : MaskableGraphic
{
/// <summary>
/// A string containing the text to be displayed.
/// </summary>
public string text
{
get { return m_text; }
set { if (m_text == value) return; m_text = old_text = value; m_inputSource = TextInputSources.String; m_havePropertiesChanged = true; m_isCalculateSizeRequired = true; m_isInputParsingRequired = true; SetVerticesDirty(); SetLayoutDirty(); }
}
[SerializeField]
[TextArea(3, 10)]
protected string m_text;
/// <summary>
///
/// </summary>
public bool isRightToLeftText
{
get { return m_isRightToLeft; }
set { if (m_isRightToLeft == value) return; m_isRightToLeft = value; m_havePropertiesChanged = true; m_isCalculateSizeRequired = true; m_isInputParsingRequired = true; SetVerticesDirty(); SetLayoutDirty(); }
}
[SerializeField]
protected bool m_isRightToLeft = false;
/// <summary>
/// The Font Asset to be assigned to this text object.
/// </summary>
public TMP_FontAsset font
{
get { return m_fontAsset; }
set { if (m_fontAsset == value) return; m_fontAsset = value; LoadFontAsset(); m_havePropertiesChanged = true; m_isCalculateSizeRequired = true; m_isInputParsingRequired = true; SetVerticesDirty(); SetLayoutDirty(); }
}
[SerializeField]
protected TMP_FontAsset m_fontAsset;
protected TMP_FontAsset m_currentFontAsset;
protected bool m_isSDFShader;
/// <summary>
/// The material to be assigned to this text object.
/// </summary>
public virtual Material fontSharedMaterial
{
get { return m_sharedMaterial; }
set { if (m_sharedMaterial == value) return; SetSharedMaterial(value); m_havePropertiesChanged = true; m_isInputParsingRequired = true; SetVerticesDirty(); SetMaterialDirty(); }
}
[SerializeField]
protected Material m_sharedMaterial;
protected Material m_currentMaterial;
protected MaterialReference[] m_materialReferences = new MaterialReference[32];
protected Dictionary<int, int> m_materialReferenceIndexLookup = new Dictionary<int, int>();
protected TMP_XmlTagStack<MaterialReference> m_materialReferenceStack = new TMP_XmlTagStack<MaterialReference>(new MaterialReference[16]);
protected int m_currentMaterialIndex;
//protected int m_sharedMaterialHashCode;
/// <summary>
/// An array containing the materials used by the text object.
/// </summary>
public virtual Material[] fontSharedMaterials
{
get { return GetSharedMaterials(); }
set { SetSharedMaterials(value); m_havePropertiesChanged = true; m_isInputParsingRequired = true; SetVerticesDirty(); SetMaterialDirty(); }
}
[SerializeField]
protected Material[] m_fontSharedMaterials;
/// <summary>
/// The material to be assigned to this text object. An instance of the material will be assigned to the object's renderer.
/// </summary>
public Material fontMaterial
{
// Return an Instance of the current material.
get { return GetMaterial(m_sharedMaterial); }
// Assign new font material
set
{
if (m_sharedMaterial != null && m_sharedMaterial.GetInstanceID() == value.GetInstanceID()) return;
m_sharedMaterial = value;
m_padding = GetPaddingForMaterial();
m_havePropertiesChanged = true;
m_isInputParsingRequired = true;
SetVerticesDirty();
SetMaterialDirty();
}
}
[SerializeField]
protected Material m_fontMaterial;
/// <summary>
/// The materials to be assigned to this text object. An instance of the materials will be assigned.
/// </summary>
public virtual Material[] fontMaterials
{
get { return GetMaterials(m_fontSharedMaterials); }
set { SetSharedMaterials(value); m_havePropertiesChanged = true; m_isInputParsingRequired = true; SetVerticesDirty(); SetMaterialDirty(); }
}
[SerializeField]
protected Material[] m_fontMaterials;
protected bool m_isMaterialDirty;
/// <summary>
/// This is the default vertex color assigned to each vertices. Color tags will override vertex colors unless the overrideColorTags is set.
/// </summary>
public override Color color
{
get { return m_fontColor; }
set { if (m_fontColor == value) return; m_havePropertiesChanged = true; m_fontColor = value; SetVerticesDirty(); }
}
//[UnityEngine.Serialization.FormerlySerializedAs("m_fontColor")] // Required for backwards compatibility with pre-Unity 4.6 releases.
[SerializeField]
protected Color32 m_fontColor32 = Color.white;
[SerializeField]
protected Color m_fontColor = Color.white;
protected static Color32 s_colorWhite = new Color32(255, 255, 255, 255);
protected Color32 m_underlineColor = s_colorWhite;
protected Color32 m_strikethroughColor = s_colorWhite;
protected Color32 m_highlightColor = s_colorWhite;
/// <summary>
/// Sets the vertex color alpha value.
/// </summary>
public float alpha
{
get { return m_fontColor.a; }
set { if (m_fontColor.a == value) return; m_fontColor.a = value; m_havePropertiesChanged = true; SetVerticesDirty(); }
}
/// <summary>
/// Determines if Vertex Color Gradient should be used
/// </summary>
/// <value><c>true</c> if enable vertex gradient; otherwise, <c>false</c>.</value>
public bool enableVertexGradient
{
get { return m_enableVertexGradient; }
set { if (m_enableVertexGradient == value) return; m_havePropertiesChanged = true; m_enableVertexGradient = value; SetVerticesDirty(); }
}
[SerializeField]
protected bool m_enableVertexGradient;
[SerializeField]
protected ColorMode m_colorMode = ColorMode.FourCornersGradient;
/// <summary>
/// Sets the vertex colors for each of the 4 vertices of the character quads.
/// </summary>
/// <value>The color gradient.</value>
public VertexGradient colorGradient
{
get { return m_fontColorGradient; }
set { m_havePropertiesChanged = true; m_fontColorGradient = value; SetVerticesDirty(); }
}
[SerializeField]
protected VertexGradient m_fontColorGradient = new VertexGradient(Color.white);
/// <summary>
/// Set the vertex colors of the 4 vertices of each character quads.
/// </summary>
public TMP_ColorGradient colorGradientPreset
{
get { return m_fontColorGradientPreset; }
set { m_havePropertiesChanged = true; m_fontColorGradientPreset = value; SetVerticesDirty(); }
}
[SerializeField]
protected TMP_ColorGradient m_fontColorGradientPreset;
/// <summary>
/// Default Sprite Asset used by the text object.
/// </summary>
public TMP_SpriteAsset spriteAsset
{
get { return m_spriteAsset; }
set { m_spriteAsset = value; m_havePropertiesChanged = true; m_isInputParsingRequired = true; m_isCalculateSizeRequired = true; SetVerticesDirty(); SetLayoutDirty(); }
}
[SerializeField]
protected TMP_SpriteAsset m_spriteAsset;
/// <summary>
/// Determines whether or not the sprite color is multiplies by the vertex color of the text.
/// </summary>
public bool tintAllSprites
{
get { return m_tintAllSprites; }
set { if (m_tintAllSprites == value) return; m_tintAllSprites = value; m_havePropertiesChanged = true; SetVerticesDirty(); }
}
[SerializeField]
protected bool m_tintAllSprites;
protected bool m_tintSprite;
protected Color32 m_spriteColor;
/// <summary>
/// This overrides the color tags forcing the vertex colors to be the default font color.
/// </summary>
public bool overrideColorTags
{
get { return m_overrideHtmlColors; }
set { if (m_overrideHtmlColors == value) return; m_havePropertiesChanged = true; m_overrideHtmlColors = value; SetVerticesDirty(); }
}
[SerializeField]
protected bool m_overrideHtmlColors = false;
/// <summary>
/// Sets the color of the _FaceColor property of the assigned material. Changing face color will result in an instance of the material.
/// </summary>
public Color32 faceColor
{
get
{
if (m_sharedMaterial == null) return m_faceColor;
m_faceColor = m_sharedMaterial.GetColor(ShaderUtilities.ID_FaceColor);
return m_faceColor;
}
set { if (m_faceColor.Compare(value)) return; SetFaceColor(value); m_havePropertiesChanged = true; m_faceColor = value; SetVerticesDirty(); SetMaterialDirty(); }
}
[SerializeField]
protected Color32 m_faceColor = Color.white;
/// <summary>
/// Sets the color of the _OutlineColor property of the assigned material. Changing outline color will result in an instance of the material.
/// </summary>
public Color32 outlineColor
{
get
{
if (m_sharedMaterial == null) return m_outlineColor;
m_outlineColor = m_sharedMaterial.GetColor(ShaderUtilities.ID_OutlineColor);
return m_outlineColor;
}
set { if (m_outlineColor.Compare(value)) return; SetOutlineColor(value); m_havePropertiesChanged = true; m_outlineColor = value; SetVerticesDirty(); }
}
[SerializeField]
protected Color32 m_outlineColor = Color.black;
/// <summary>
/// Sets the thickness of the outline of the font. Setting this value will result in an instance of the material.
/// </summary>
public float outlineWidth
{
get
{
if (m_sharedMaterial == null) return m_outlineWidth;
m_outlineWidth = m_sharedMaterial.GetFloat(ShaderUtilities.ID_OutlineWidth);
return m_outlineWidth;
}
set { if (m_outlineWidth == value) return; SetOutlineThickness(value); m_havePropertiesChanged = true; m_outlineWidth = value; SetVerticesDirty(); }
}
protected float m_outlineWidth = 0.0f;
/// <summary>
/// The point size of the font.
/// </summary>
public float fontSize
{
get { return m_fontSize; }
set { if (m_fontSize == value) return; m_havePropertiesChanged = true; m_isCalculateSizeRequired = true; m_fontSize = value; if (!m_enableAutoSizing) m_fontSizeBase = m_fontSize; SetVerticesDirty(); SetLayoutDirty(); }
}
[SerializeField]
protected float m_fontSize = 36; // Font Size
protected float m_currentFontSize; // Temporary Font Size affected by tags
[SerializeField]
protected float m_fontSizeBase = 36;
protected TMP_XmlTagStack<float> m_sizeStack = new TMP_XmlTagStack<float>(new float[16]);
/// <summary>
/// The scale of the current text.
/// </summary>
public float fontScale
{
get { return m_fontScale; }
}
/// <summary>
/// Control the weight of the font if an alternative font asset is assigned for the given weight in the font asset editor.
/// </summary>
public int fontWeight
{
get { return m_fontWeight; }
set { if (m_fontWeight == value) return; m_fontWeight = value; m_havePropertiesChanged = true; m_isCalculateSizeRequired = true; m_isInputParsingRequired = true; SetVerticesDirty(); SetLayoutDirty(); }
}
[SerializeField]
protected int m_fontWeight = 400;
protected int m_fontWeightInternal;
protected TMP_XmlTagStack<int> m_fontWeightStack = new TMP_XmlTagStack<int>(new int[16]);
/// <summary>
///
/// </summary>
public float pixelsPerUnit
{
get
{
var localCanvas = canvas;
if (!localCanvas)
return 1;
// For dynamic fonts, ensure we use one pixel per pixel on the screen.
if (!font)
return localCanvas.scaleFactor;
// For non-dynamic fonts, calculate pixels per unit based on specified font size relative to font object's own font size.
if (m_currentFontAsset == null || m_currentFontAsset.fontInfo.PointSize <= 0 || m_fontSize <= 0)
return 1;
return m_fontSize / m_currentFontAsset.fontInfo.PointSize;
}
}
/// <summary>
/// Enable text auto-sizing
/// </summary>
public bool enableAutoSizing
{
get { return m_enableAutoSizing; }
set { if (m_enableAutoSizing == value) return; m_enableAutoSizing = value; SetVerticesDirty(); SetLayoutDirty(); }
}
[SerializeField]
protected bool m_enableAutoSizing;
protected float m_maxFontSize; // Used in conjunction with auto-sizing
protected float m_minFontSize; // Used in conjunction with auto-sizing
/// <summary>
/// Minimum point size of the font when text auto-sizing is enabled.
/// </summary>
public float fontSizeMin
{
get { return m_fontSizeMin; }
set { if (m_fontSizeMin == value) return; m_fontSizeMin = value; SetVerticesDirty(); SetLayoutDirty(); }
}
[SerializeField]
protected float m_fontSizeMin = 0; // Text Auto Sizing Min Font Size.
/// <summary>
/// Maximum point size of the font when text auto-sizing is enabled.
/// </summary>
public float fontSizeMax
{
get { return m_fontSizeMax; }
set { if (m_fontSizeMax == value) return; m_fontSizeMax = value; SetVerticesDirty(); SetLayoutDirty(); }
}
[SerializeField]
protected float m_fontSizeMax = 0; // Text Auto Sizing Max Font Size.
/// <summary>
/// The style of the text
/// </summary>
public FontStyles fontStyle
{
get { return m_fontStyle; }
set { if (m_fontStyle == value) return; m_fontStyle = value; m_havePropertiesChanged = true; m_isCalculateSizeRequired = true; m_isInputParsingRequired = true; SetVerticesDirty(); SetLayoutDirty(); }
}
[SerializeField]
protected FontStyles m_fontStyle = FontStyles.Normal;
protected FontStyles m_style = FontStyles.Normal;
protected TMP_BasicXmlTagStack m_fontStyleStack;
/// <summary>
/// Property used in conjunction with padding calculation for the geometry.
/// </summary>
public bool isUsingBold { get { return m_isUsingBold; } }
protected bool m_isUsingBold = false; // Used to ensure GetPadding & Ratios take into consideration bold characters.
/// <summary>
/// Text alignment options
/// </summary>
public TextAlignmentOptions alignment
{
get { return m_textAlignment; }
set { if (m_textAlignment == value) return; m_havePropertiesChanged = true; m_textAlignment = value; SetVerticesDirty(); }
}
[SerializeField]
[UnityEngine.Serialization.FormerlySerializedAs("m_lineJustification")]
protected TextAlignmentOptions m_textAlignment = TextAlignmentOptions.TopLeft;
protected TextAlignmentOptions m_lineJustification;
protected TMP_XmlTagStack<TextAlignmentOptions> m_lineJustificationStack = new TMP_XmlTagStack<TextAlignmentOptions>(new TextAlignmentOptions[16]);
protected Vector3[] m_textContainerLocalCorners = new Vector3[4];
[SerializeField]
protected bool m_isAlignmentEnumConverted;
/// <summary>
/// Use the extents of the text geometry for alignment instead of font metrics.
/// </summary>
//public bool alignByGeometry
//{
// get { return m_alignByGeometry; }
// set { if (m_alignByGeometry == value) return; m_havePropertiesChanged = true; m_alignByGeometry = value; SetVerticesDirty(); }
//}
//[SerializeField]
//protected bool m_alignByGeometry;
/// <summary>
/// The amount of additional spacing between characters.
/// </summary>
public float characterSpacing
{
get { return m_characterSpacing; }
set { if (m_characterSpacing == value) return; m_havePropertiesChanged = true; m_isCalculateSizeRequired = true; m_characterSpacing = value; SetVerticesDirty(); SetLayoutDirty(); }
}
[SerializeField]
protected float m_characterSpacing = 0;
protected float m_cSpacing = 0;
protected float m_monoSpacing = 0;
/// <summary>
/// The amount of additional spacing between words.
/// </summary>
public float wordSpacing
{
get { return m_wordSpacing; }
set { if (m_wordSpacing == value) return; m_havePropertiesChanged = true; m_isCalculateSizeRequired = true; m_wordSpacing = value; SetVerticesDirty(); SetLayoutDirty(); }
}
[SerializeField]
protected float m_wordSpacing = 0;
/// <summary>
/// The amount of additional spacing to add between each lines of text.
/// </summary>
public float lineSpacing
{
get { return m_lineSpacing; }
set { if (m_lineSpacing == value) return; m_havePropertiesChanged = true; m_isCalculateSizeRequired = true; m_lineSpacing = value; SetVerticesDirty(); SetLayoutDirty(); }
}
[SerializeField]
protected float m_lineSpacing = 0;
protected float m_lineSpacingDelta = 0; // Used with Text Auto Sizing feature
protected float m_lineHeight = TMP_Math.FLOAT_UNSET; // Used with the <line-height=xx.x> tag.
/// <summary>
/// The amount of potential line spacing adjustment before text auto sizing kicks in.
/// </summary>
public float lineSpacingAdjustment
{
get { return m_lineSpacingMax; }
set { if (m_lineSpacingMax == value) return; m_havePropertiesChanged = true; m_isCalculateSizeRequired = true; m_lineSpacingMax = value; SetVerticesDirty(); SetLayoutDirty(); }
}
[SerializeField]
protected float m_lineSpacingMax = 0; // Text Auto Sizing Max Line spacing reduction.
//protected bool m_forceLineBreak;
/// <summary>
/// The amount of additional spacing to add between each lines of text.
/// </summary>
public float paragraphSpacing
{
get { return m_paragraphSpacing; }
set { if (m_paragraphSpacing == value) return; m_havePropertiesChanged = true; m_isCalculateSizeRequired = true; m_paragraphSpacing = value; SetVerticesDirty(); SetLayoutDirty(); }
}
[SerializeField]
protected float m_paragraphSpacing = 0;
/// <summary>
/// Percentage the width of characters can be adjusted before text auto-sizing begins to reduce the point size.
/// </summary>
public float characterWidthAdjustment
{
get { return m_charWidthMaxAdj; }
set { if (m_charWidthMaxAdj == value) return; m_havePropertiesChanged = true; m_isCalculateSizeRequired = true; m_charWidthMaxAdj = value; SetVerticesDirty(); SetLayoutDirty(); }
}
[SerializeField]
protected float m_charWidthMaxAdj = 0f; // Text Auto Sizing Max Character Width reduction.
protected float m_charWidthAdjDelta = 0;
/// <summary>
/// Controls whether or not word wrapping is applied. When disabled, the text will be displayed on a single line.
/// </summary>
public bool enableWordWrapping
{
get { return m_enableWordWrapping; }
set { if (m_enableWordWrapping == value) return; m_havePropertiesChanged = true; m_isInputParsingRequired = true; m_isCalculateSizeRequired = true; m_enableWordWrapping = value; SetVerticesDirty(); SetLayoutDirty(); }
}
[SerializeField]
protected bool m_enableWordWrapping = false;
protected bool m_isCharacterWrappingEnabled = false;
protected bool m_isNonBreakingSpace = false;
protected bool m_isIgnoringAlignment;
/// <summary>
/// Controls the blending between using character and word spacing to fill-in the space for justified text.
/// </summary>
public float wordWrappingRatios
{
get { return m_wordWrappingRatios; }
set { if (m_wordWrappingRatios == value) return; m_wordWrappingRatios = value; m_havePropertiesChanged = true; m_isCalculateSizeRequired = true; SetVerticesDirty(); SetLayoutDirty(); }
}
[SerializeField]
protected float m_wordWrappingRatios = 0.4f; // Controls word wrapping ratios between word or characters.
/// <summary>
///
/// </summary>
//public bool enableAdaptiveJustification
//{
// get { return m_enableAdaptiveJustification; }
// set { if (m_enableAdaptiveJustification == value) return; m_enableAdaptiveJustification = value; m_havePropertiesChanged = true; m_isCalculateSizeRequired = true; SetVerticesDirty(); SetLayoutDirty(); }
//}
//[SerializeField]
//protected bool m_enableAdaptiveJustification;
//protected float m_adaptiveJustificationThreshold = 10.0f;
/// <summary>
/// Controls the Text Overflow Mode
/// </summary>
public TextOverflowModes overflowMode
{
get { return m_overflowMode; }
set { if (m_overflowMode == value) return; m_overflowMode = value; m_havePropertiesChanged = true; m_isCalculateSizeRequired = true; SetVerticesDirty(); SetLayoutDirty(); }
}
[SerializeField]
protected TextOverflowModes m_overflowMode = TextOverflowModes.Overflow;
/// <summary>
/// Indicates if the text exceeds the vertical bounds of its text container.
/// </summary>
public bool isTextOverflowing
{
get { if (m_firstOverflowCharacterIndex != -1) return true; return false; }
}
/// <summary>
/// The first character which exceeds the vertical bounds of its text container.
/// </summary>
public int firstOverflowCharacterIndex
{
get { return m_firstOverflowCharacterIndex; }
}
[SerializeField]
protected int m_firstOverflowCharacterIndex = -1;
/// <summary>
/// The linked text component used for flowing the text from one text component to another.
/// </summary>
public TMP_Text linkedTextComponent
{
get { return m_linkedTextComponent; }
set
{
if (m_linkedTextComponent != value)
{
// Release previously linked text component.
if (m_linkedTextComponent != null)
{
m_linkedTextComponent.overflowMode = TextOverflowModes.Overflow;
m_linkedTextComponent.linkedTextComponent = null;
m_linkedTextComponent.isLinkedTextComponent = false;
}
m_linkedTextComponent = value;
if (m_linkedTextComponent != null)
m_linkedTextComponent.isLinkedTextComponent = true;
}
m_havePropertiesChanged = true;
m_isCalculateSizeRequired = true;
SetVerticesDirty();
SetLayoutDirty();
}
}
[SerializeField]
protected TMP_Text m_linkedTextComponent;
/// <summary>
/// Indicates whether this text component is linked to another.
/// </summary>
public bool isLinkedTextComponent
{
get { return m_isLinkedTextComponent; }
set
{
m_isLinkedTextComponent = value;
if (m_isLinkedTextComponent == false)
m_firstVisibleCharacter = 0;
m_havePropertiesChanged = true;
m_isCalculateSizeRequired = true;
SetVerticesDirty();
SetLayoutDirty();
}
}
[SerializeField]
protected bool m_isLinkedTextComponent;
/// <summary>
/// Property indicating whether the text is Truncated or using Ellipsis.
/// </summary>
public bool isTextTruncated { get { return m_isTextTruncated; } }
[SerializeField]
protected bool m_isTextTruncated;
/// <summary>
/// Determines if kerning is enabled or disabled.
/// </summary>
public bool enableKerning
{
get { return m_enableKerning; }
set { if (m_enableKerning == value) return; m_havePropertiesChanged = true; m_isCalculateSizeRequired = true; m_enableKerning = value; SetVerticesDirty(); SetLayoutDirty(); }
}
[SerializeField]
protected bool m_enableKerning;
/// <summary>
/// Adds extra padding around each character. This may be necessary when the displayed text is very small to prevent clipping.
/// </summary>
public bool extraPadding
{
get { return m_enableExtraPadding; }
set { if (m_enableExtraPadding == value) return; m_havePropertiesChanged = true; m_enableExtraPadding = value; UpdateMeshPadding(); /* m_isCalculateSizeRequired = true;*/ SetVerticesDirty(); /* SetLayoutDirty();*/ }
}
[SerializeField]
protected bool m_enableExtraPadding = false;
[SerializeField]
protected bool checkPaddingRequired;
/// <summary>
/// Enables or Disables Rich Text Tags
/// </summary>
public bool richText
{
get { return m_isRichText; }
set { if (m_isRichText == value) return; m_isRichText = value; m_havePropertiesChanged = true; m_isCalculateSizeRequired = true; m_isInputParsingRequired = true; SetVerticesDirty(); SetLayoutDirty(); }
}
[SerializeField]
protected bool m_isRichText = true; // Used to enable or disable Rich Text.
/// <summary>
/// Enables or Disables parsing of CTRL characters in input text.
/// </summary>
public bool parseCtrlCharacters
{
get { return m_parseCtrlCharacters; }
set { if (m_parseCtrlCharacters == value) return; m_parseCtrlCharacters = value; m_havePropertiesChanged = true; m_isCalculateSizeRequired = true; m_isInputParsingRequired = true; SetVerticesDirty(); SetLayoutDirty(); }
}
[SerializeField]
protected bool m_parseCtrlCharacters = true;
/// <summary>
/// Sets the RenderQueue along with Ztest to force the text to be drawn last and on top of scene elements.
/// </summary>
public bool isOverlay
{
get { return m_isOverlay; }
set { if (m_isOverlay == value) return; m_isOverlay = value; SetShaderDepth(); m_havePropertiesChanged = true; SetVerticesDirty(); }
}
protected bool m_isOverlay = false;
/// <summary>
/// Sets Perspective Correction to Zero for Orthographic Camera mode & 0.875f for Perspective Camera mode.
/// </summary>
public bool isOrthographic
{
get { return m_isOrthographic; }
set { if (m_isOrthographic == value) return; m_havePropertiesChanged = true; m_isOrthographic = value; SetVerticesDirty(); }
}
[SerializeField]
protected bool m_isOrthographic = false;
/// <summary>
/// Sets the culling on the shaders. Note changing this value will result in an instance of the material.
/// </summary>
public bool enableCulling
{
get { return m_isCullingEnabled; }
set { if (m_isCullingEnabled == value) return; m_isCullingEnabled = value; SetCulling(); m_havePropertiesChanged = true; }
}
[SerializeField]
protected bool m_isCullingEnabled = false;
/// <summary>
/// Controls whether or not the text object will be culled when using a 2D Rect Mask.
/// </summary>
public bool ignoreRectMaskCulling
{
get { return m_ignoreRectMaskCulling; }
set { if (m_ignoreRectMaskCulling == value) return; m_ignoreRectMaskCulling = value; m_havePropertiesChanged = true; }
}
[SerializeField]
protected bool m_ignoreRectMaskCulling;
/// <summary>
/// Forces objects that are not visible to get refreshed.
/// </summary>
public bool ignoreVisibility
{
get { return m_ignoreCulling; }
set { if (m_ignoreCulling == value) return; m_havePropertiesChanged = true; m_ignoreCulling = value; }
}
[SerializeField]
protected bool m_ignoreCulling = true; // Not implemented yet.
/// <summary>
/// Controls how the face and outline textures will be applied to the text object.
/// </summary>
public TextureMappingOptions horizontalMapping
{
get { return m_horizontalMapping; }
set { if (m_horizontalMapping == value) return; m_havePropertiesChanged = true; m_horizontalMapping = value; SetVerticesDirty(); }
}
[SerializeField]
protected TextureMappingOptions m_horizontalMapping = TextureMappingOptions.Character;
/// <summary>
/// Controls how the face and outline textures will be applied to the text object.
/// </summary>
public TextureMappingOptions verticalMapping
{
get { return m_verticalMapping; }
set { if (m_verticalMapping == value) return; m_havePropertiesChanged = true; m_verticalMapping = value; SetVerticesDirty(); }
}
[SerializeField]
protected TextureMappingOptions m_verticalMapping = TextureMappingOptions.Character;
/// <summary>
/// Controls the UV Offset for the various texture mapping mode on the text object.
/// </summary>
//public Vector2 mappingUvOffset
//{
// get { return m_uvOffset; }
// set { if (m_uvOffset == value) return; m_havePropertiesChanged = true; m_uvOffset = value; SetVerticesDirty(); }
//}
//[SerializeField]
//protected Vector2 m_uvOffset = Vector2.zero; // Used to offset UV on Texturing
/// <summary>
/// Controls the horizontal offset of the UV of the texture mapping mode for each line of the text object.
/// </summary>
public float mappingUvLineOffset
{
get { return m_uvLineOffset; }
set { if (m_uvLineOffset == value) return; m_havePropertiesChanged = true; m_uvLineOffset = value; SetVerticesDirty(); }
}
[SerializeField]
protected float m_uvLineOffset = 0.0f; // Used for UV line offset per line
/// <summary>
/// Determines if the Mesh will be rendered.
/// </summary>
public TextRenderFlags renderMode
{
get { return m_renderMode; }
set { if (m_renderMode == value) return; m_renderMode = value; m_havePropertiesChanged = true; }
}
protected TextRenderFlags m_renderMode = TextRenderFlags.Render;
/// <summary>
/// Determines the sorting order of the geometry of the text object.
/// </summary>
public VertexSortingOrder geometrySortingOrder
{
get { return m_geometrySortingOrder; }
set { m_geometrySortingOrder = value; m_havePropertiesChanged = true; SetVerticesDirty(); }
}
[SerializeField]
protected VertexSortingOrder m_geometrySortingOrder;
/// <summary>
/// The first character which should be made visible in conjunction with the Text Overflow Linked mode.
/// </summary>
public int firstVisibleCharacter
{
get { return m_firstVisibleCharacter; }
set { if (m_firstVisibleCharacter == value) return; m_havePropertiesChanged = true; m_firstVisibleCharacter = value; SetVerticesDirty(); }
}
[SerializeField]
protected int m_firstVisibleCharacter;
/// <summary>
/// Allows to control how many characters are visible from the input.
/// </summary>
public int maxVisibleCharacters
{
get { return m_maxVisibleCharacters; }
set { if (m_maxVisibleCharacters == value) return; m_havePropertiesChanged = true; m_maxVisibleCharacters = value; SetVerticesDirty(); }
}
protected int m_maxVisibleCharacters = 99999;
/// <summary>
/// Allows to control how many words are visible from the input.
/// </summary>
public int maxVisibleWords
{
get { return m_maxVisibleWords; }
set { if (m_maxVisibleWords == value) return; m_havePropertiesChanged = true; m_maxVisibleWords = value; SetVerticesDirty(); }
}
protected int m_maxVisibleWords = 99999;
/// <summary>
/// Allows control over how many lines of text are displayed.
/// </summary>
public int maxVisibleLines
{
get { return m_maxVisibleLines; }
set { if (m_maxVisibleLines == value) return; m_havePropertiesChanged = true; m_isInputParsingRequired = true; m_maxVisibleLines = value; SetVerticesDirty(); }
}
protected int m_maxVisibleLines = 99999;
/// <summary>
/// Determines if the text's vertical alignment will be adjusted based on visible descender of the text.
/// </summary>
public bool useMaxVisibleDescender
{
get { return m_useMaxVisibleDescender; }
set { if (m_useMaxVisibleDescender == value) return; m_havePropertiesChanged = true; m_isInputParsingRequired = true; SetVerticesDirty(); }
}
[SerializeField]
protected bool m_useMaxVisibleDescender = true;
/// <summary>
/// Controls which page of text is shown
/// </summary>
public int pageToDisplay
{
get { return m_pageToDisplay; }
set { if (m_pageToDisplay == value) return; m_havePropertiesChanged = true; m_pageToDisplay = value; SetVerticesDirty(); }
}
[SerializeField]
protected int m_pageToDisplay = 1;
protected bool m_isNewPage = false;
/// <summary>
/// The margins of the text object.
/// </summary>
public virtual Vector4 margin
{
get { return m_margin; }
set { if (m_margin == value) return; m_margin = value; ComputeMarginSize(); m_havePropertiesChanged = true; SetVerticesDirty(); }
}
[SerializeField]
protected Vector4 m_margin = new Vector4(0, 0, 0, 0);
protected float m_marginLeft;
protected float m_marginRight;
protected float m_marginWidth; // Width of the RectTransform minus left and right margins.
protected float m_marginHeight; // Height of the RectTransform minus top and bottom margins.
protected float m_width = -1;
/// <summary>
/// Returns data about the text object which includes information about each character, word, line, link, etc.
/// </summary>
public TMP_TextInfo textInfo
{
get { return m_textInfo; }
}
[SerializeField]
protected TMP_TextInfo m_textInfo; // Class which holds information about the Text object such as characters, lines, mesh data as well as metrics.
/// <summary>
/// Property tracking if any of the text properties have changed. Flag is set before the text is regenerated.
/// </summary>
public bool havePropertiesChanged
{
get { return m_havePropertiesChanged; }
set { if (m_havePropertiesChanged == value) return; m_havePropertiesChanged = value; m_isInputParsingRequired = true; SetAllDirty(); }
}
[SerializeField]
protected bool m_havePropertiesChanged; // Used to track when properties of the text object have changed.
/// <summary>
/// Property to handle legacy animation component.
/// </summary>
public bool isUsingLegacyAnimationComponent
{
get { return m_isUsingLegacyAnimationComponent; }
set { m_isUsingLegacyAnimationComponent = value; }
}
[SerializeField]
protected bool m_isUsingLegacyAnimationComponent;
/// <summary>
/// Returns are reference to the Transform
/// </summary>
public new Transform transform
{
get
{
if (m_transform == null)
m_transform = GetComponent<Transform>();
return m_transform;
}
}
protected Transform m_transform;
/// <summary>
/// Returns are reference to the RectTransform
/// </summary>
public new RectTransform rectTransform
{
get
{
if (m_rectTransform == null)
m_rectTransform = GetComponent<RectTransform>();
return m_rectTransform;
}
}
protected RectTransform m_rectTransform;
/// <summary>
/// Enables control over setting the size of the text container to match the text object.
/// </summary>
public virtual bool autoSizeTextContainer
{
get;
set;
}
protected bool m_autoSizeTextContainer;
/// <summary>
/// The mesh used by the font asset and material assigned to the text object.
/// </summary>
public virtual Mesh mesh
{
get { return m_mesh; }
}
protected Mesh m_mesh;
/// <summary>
/// Determines if the geometry of the characters will be quads or volumetric (cubes).
/// </summary>
public bool isVolumetricText
{
get { return m_isVolumetricText; }
set { if (m_isVolumetricText == value) return; m_havePropertiesChanged = value; m_textInfo.ResetVertexLayout(value); m_isInputParsingRequired = true; SetVerticesDirty(); SetLayoutDirty(); }
}
[SerializeField]
protected bool m_isVolumetricText;
/// <summary>
/// Returns the bounds of the mesh of the text object in world space.
/// </summary>
public Bounds bounds
{
get
{
if (m_mesh == null) return new Bounds();
return GetCompoundBounds();
}
}
/// <summary>
/// Returns the bounds of the text of the text object.
/// </summary>
public Bounds textBounds
{
get
{
if (m_textInfo == null) return new Bounds();
return GetTextBounds();
}
}
// *** Unity Event Handling ***
//[Serializable]
//public class TextChangedEvent : UnityEvent { }
///// <summary>
///// Event delegate triggered when text has changed and been rendered.
///// </summary>
//public TextChangedEvent onTextChanged
//{
// get { return m_OnTextChanged; }
// set { m_OnTextChanged = value; }
//}
//[SerializeField]
//private TextChangedEvent m_OnTextChanged = new TextChangedEvent();
//protected void SendOnTextChanged()
//{
// if (onTextChanged != null)
// onTextChanged.Invoke();
//}
// *** SPECIAL COMPONENTS ***
/// <summary>
/// Component used to control wrapping of text following some arbitrary shape.
/// </summary>
//public MarginShaper marginShaper
//{
// get
// {
// if (m_marginShaper == null) m_marginShaper = GetComponent<MarginShaper>();
// return m_marginShaper;
// }
//}
//[SerializeField]
//protected MarginShaper m_marginShaper;
/// <summary>
/// Component used to control and animate sprites in the text object.
/// </summary>
protected TMP_SpriteAnimator spriteAnimator
{
get
{
if (m_spriteAnimator == null)
{
m_spriteAnimator = GetComponent<TMP_SpriteAnimator>();
if (m_spriteAnimator == null) m_spriteAnimator = gameObject.AddComponent<TMP_SpriteAnimator>();
}
return m_spriteAnimator;
}
}
[SerializeField]
protected TMP_SpriteAnimator m_spriteAnimator;
/// <summary>
///
/// </summary>
//public TMP_TextShaper textShaper
//{
// get
// {
// if (m_textShaper == null)
// m_textShaper = GetComponent<TMP_TextShaper>();
// return m_textShaper;
// }
//}
//[SerializeField]
//protected TMP_TextShaper m_textShaper;
// *** PROPERTIES RELATED TO UNITY LAYOUT SYSTEM ***
/// <summary>
///
/// </summary>
public float flexibleHeight { get { return m_flexibleHeight; } }
protected float m_flexibleHeight = -1f;
/// <summary>
///
/// </summary>
public float flexibleWidth { get { return m_flexibleWidth; } }
protected float m_flexibleWidth = -1f;
/// <summary>
///
/// </summary>
public float minWidth { get { return m_minWidth; } }
protected float m_minWidth;
/// <summary>
///
/// </summary>
public float minHeight { get { return m_minHeight; } }
protected float m_minHeight;
/// <summary>
///
/// </summary>
public float maxWidth { get { return m_maxWidth; } }
protected float m_maxWidth;
/// <summary>
///
/// </summary>
public float maxHeight { get { return m_maxHeight; } }
protected float m_maxHeight;
/// <summary>
///
/// </summary>
protected LayoutElement layoutElement
{
get
{
if (m_LayoutElement == null)
{
m_LayoutElement = GetComponent<LayoutElement>();
}
return m_LayoutElement;
}
}
protected LayoutElement m_LayoutElement;
/// <summary>
/// Computed preferred width of the text object.
/// </summary>
public virtual float preferredWidth { get { if (!m_isPreferredWidthDirty) return m_preferredWidth; m_preferredWidth = GetPreferredWidth(); return m_preferredWidth; } }
protected float m_preferredWidth;
protected float m_renderedWidth;
protected bool m_isPreferredWidthDirty;
/// <summary>
/// Computed preferred height of the text object.
/// </summary>
public virtual float preferredHeight { get { if (!m_isPreferredHeightDirty) return m_preferredHeight; m_preferredHeight = GetPreferredHeight(); return m_preferredHeight; } }
protected float m_preferredHeight;
protected float m_renderedHeight;
protected bool m_isPreferredHeightDirty;
protected bool m_isCalculatingPreferredValues;
private int m_recursiveCount;
/// <summary>
/// Compute the rendered width of the text object.
/// </summary>
public virtual float renderedWidth { get { return GetRenderedWidth(); } }
/// <summary>
/// Compute the rendered height of the text object.
/// </summary>
public virtual float renderedHeight { get { return GetRenderedHeight(); } }
/// <summary>
///
/// </summary>
public int layoutPriority { get { return m_layoutPriority; } }
protected int m_layoutPriority = 0;
protected bool m_isCalculateSizeRequired = false;
protected bool m_isLayoutDirty;
protected bool m_verticesAlreadyDirty;
protected bool m_layoutAlreadyDirty;
protected bool m_isAwake;
protected bool m_isWaitingOnResourceLoad;
[SerializeField]
protected bool m_isInputParsingRequired = false; // Used to determine if the input text needs to be re-parsed.
// Protected Fields
protected enum TextInputSources { Text = 0, SetText = 1, SetCharArray = 2, String = 3 };
[SerializeField]
protected TextInputSources m_inputSource;
protected string old_text; // Used by SetText to determine if the text has changed.
//protected float old_arg0, old_arg1, old_arg2; // Used by SetText to determine if the args have changed.
protected float m_fontScale; // Scaling of the font based on Atlas true Font Size and Rendered Font Size.
protected float m_fontScaleMultiplier; // Used for handling of superscript and subscript.
protected char[] m_htmlTag = new char[128]; // Maximum length of rich text tag. This is preallocated to avoid GC.
protected XML_TagAttribute[] m_xmlAttribute = new XML_TagAttribute[8];
protected float[] m_attributeParameterValues = new float[16];
protected float tag_LineIndent = 0;
protected float tag_Indent = 0;
protected TMP_XmlTagStack<float> m_indentStack = new TMP_XmlTagStack<float>(new float[16]);
protected bool tag_NoParsing;
//protected TMP_LinkInfo tag_LinkInfo = new TMP_LinkInfo();
protected bool m_isParsingText;
protected Matrix4x4 m_FXMatrix;
protected bool m_isFXMatrixSet;
protected int[] m_char_buffer; // This array holds the characters to be processed by GenerateMesh();
private TMP_CharacterInfo[] m_internalCharacterInfo; // Used by functions to calculate preferred values.
protected char[] m_input_CharArray = new char[256]; // This array hold the characters from the SetText();
private int m_charArray_Length = 0;
protected int m_totalCharacterCount;
// Structures used to save the state of the text layout in conjunction with line breaking / word wrapping.
protected WordWrapState m_SavedWordWrapState = new WordWrapState();
protected WordWrapState m_SavedLineState = new WordWrapState();
//protected WordWrapState m_SavedAlignment = new WordWrapState ();
// Fields whose state is saved in conjunction with text parsing and word wrapping.
protected int m_characterCount;
//protected int m_visibleCharacterCount;
//protected int m_visibleSpriteCount;
protected int m_firstCharacterOfLine;
protected int m_firstVisibleCharacterOfLine;
protected int m_lastCharacterOfLine;
protected int m_lastVisibleCharacterOfLine;
protected int m_lineNumber;
protected int m_lineVisibleCharacterCount;
protected int m_pageNumber;
protected float m_maxAscender;
protected float m_maxCapHeight;
protected float m_maxDescender;
protected float m_maxLineAscender;
protected float m_maxLineDescender;
protected float m_startOfLineAscender;
//protected float m_maxFontScale;
protected float m_lineOffset;
protected Extents m_meshExtents;
// Fields used for vertex colors
protected Color32 m_htmlColor = new Color(255, 255, 255, 128);
protected TMP_XmlTagStack<Color32> m_colorStack = new TMP_XmlTagStack<Color32>(new Color32[16]);
protected TMP_XmlTagStack<Color32> m_underlineColorStack = new TMP_XmlTagStack<Color32>(new Color32[16]);
protected TMP_XmlTagStack<Color32> m_strikethroughColorStack = new TMP_XmlTagStack<Color32>(new Color32[16]);
protected TMP_XmlTagStack<Color32> m_highlightColorStack = new TMP_XmlTagStack<Color32>(new Color32[16]);
protected TMP_ColorGradient m_colorGradientPreset;
protected TMP_XmlTagStack<TMP_ColorGradient> m_colorGradientStack = new TMP_XmlTagStack<TMP_ColorGradient>(new TMP_ColorGradient[16]);
protected float m_tabSpacing = 0;
protected float m_spacing = 0;
//protected bool IsRectTransformDriven;
// STYLE TAGS
protected TMP_XmlTagStack<int> m_styleStack = new TMP_XmlTagStack<int>(new int[16]);
protected TMP_XmlTagStack<int> m_actionStack = new TMP_XmlTagStack<int>(new int[16]);
protected float m_padding = 0;
protected float m_baselineOffset; // Used for superscript and subscript.
protected TMP_XmlTagStack<float> m_baselineOffsetStack = new TMP_XmlTagStack<float>(new float[16]);
protected float m_xAdvance; // Tracks x advancement from character to character.
protected TMP_TextElementType m_textElementType;
protected TMP_TextElement m_cached_TextElement; // Glyph / Character information is cached into this variable which is faster than having to fetch from the Dictionary multiple times.
protected TMP_Glyph m_cached_Underline_GlyphInfo; // Same as above but for the underline character which is used for Underline.
protected TMP_Glyph m_cached_Ellipsis_GlyphInfo;
protected TMP_SpriteAsset m_defaultSpriteAsset;
protected TMP_SpriteAsset m_currentSpriteAsset;
protected int m_spriteCount = 0;
protected int m_spriteIndex;
protected int m_spriteAnimationID;
//protected TMP_XmlTagStack<int> m_spriteAnimationStack = new TMP_XmlTagStack<int>(new int[16]);
/// <summary>
/// Method which derived classes need to override to load Font Assets.
/// </summary>
protected virtual void LoadFontAsset() { }
/// <summary>
/// Function called internally when a new shared material is assigned via the fontSharedMaterial property.
/// </summary>
/// <param name="mat"></param>
protected virtual void SetSharedMaterial(Material mat) { }
/// <summary>
/// Function called internally when a new material is assigned via the fontMaterial property.
/// </summary>
protected virtual Material GetMaterial(Material mat) { return null; }
/// <summary>
/// Function called internally when assigning a new base material.
/// </summary>
/// <param name="mat"></param>
protected virtual void SetFontBaseMaterial(Material mat) { }
/// <summary>
/// Method which returns an array containing the materials used by the text object.
/// </summary>
/// <returns></returns>
protected virtual Material[] GetSharedMaterials() { return null; }
/// <summary>
///
/// </summary>
protected virtual void SetSharedMaterials(Material[] materials) { }
/// <summary>
/// Method returning instances of the materials used by the text object.
/// </summary>
/// <returns></returns>
protected virtual Material[] GetMaterials(Material[] mats) { return null; }
/// <summary>
/// Method to set the materials of the text and sub text objects.
/// </summary>
/// <param name="mats"></param>
//protected virtual void SetMaterials (Material[] mats) { }
/// <summary>
/// Function used to create an instance of the material
/// </summary>
/// <param name="source"></param>
/// <returns></returns>
protected virtual Material CreateMaterialInstance(Material source)
{
Material mat = new Material(source);
mat.shaderKeywords = source.shaderKeywords;
mat.name += " (Instance)";
return mat;
}
protected void SetVertexColorGradient(TMP_ColorGradient gradient)
{
if (gradient == null) return;
m_fontColorGradient.bottomLeft = gradient.bottomLeft;
m_fontColorGradient.bottomRight = gradient.bottomRight;
m_fontColorGradient.topLeft = gradient.topLeft;
m_fontColorGradient.topRight = gradient.topRight;
SetVerticesDirty();
}
/// <summary>
/// Function to control the sorting of the geometry of the text object.
/// </summary>
protected void SetTextSortingOrder(VertexSortingOrder order)
{
}
/// <summary>
/// Function to sort the geometry of the text object in accordance to the provided order.
/// </summary>
/// <param name="order"></param>
protected void SetTextSortingOrder(int[] order)
{
}
/// <summary>
/// Function called internally to set the face color of the material. This will results in an instance of the material.
/// </summary>
/// <param name="color"></param>
protected virtual void SetFaceColor(Color32 color) { }
/// <summary>
/// Function called internally to set the outline color of the material. This will results in an instance of the material.
/// </summary>
/// <param name="color"></param>
protected virtual void SetOutlineColor(Color32 color) { }
/// <summary>
/// Function called internally to set the outline thickness property of the material. This will results in an instance of the material.
/// </summary>
/// <param name="thickness"></param>
protected virtual void SetOutlineThickness(float thickness) { }
/// <summary>
/// Set the Render Queue and ZTest mode on the current material
/// </summary>
protected virtual void SetShaderDepth() { }
/// <summary>
/// Set the culling mode on the material.
/// </summary>
protected virtual void SetCulling() { }
/// <summary>
/// Get the padding value for the currently assigned material
/// </summary>
/// <returns></returns>
protected virtual float GetPaddingForMaterial() { return 0; }
/// <summary>
/// Get the padding value for the given material
/// </summary>
/// <returns></returns>
protected virtual float GetPaddingForMaterial(Material mat) { return 0; }
/// <summary>
/// Method to return the local corners of the Text Container or RectTransform.
/// </summary>
/// <returns></returns>
protected virtual Vector3[] GetTextContainerLocalCorners() { return null; }
// PUBLIC FUNCTIONS
protected bool m_ignoreActiveState;
/// <summary>
/// Function to force the regeneration of the text object.
/// </summary>
public virtual void ForceMeshUpdate() { }
/// <summary>
/// Method used for resetting vertex layout when switching to and from Volumetric Text mode.
/// </summary>
/// <param name="updateMesh"></param>
//protected virtual void ResetVertexLayout() { }
/// <summary>
/// Function to force the regeneration of the text object.
/// </summary>
/// <param name="ignoreActiveState">If set to true, the text object will be regenerated regardless of is active state.</param>
public virtual void ForceMeshUpdate(bool ignoreActiveState) { }
/// <summary>
/// Internal function used by the Text Input Field to populate TMP_TextInfo data.
/// </summary>
internal void SetTextInternal(string text)
{
m_text = text;
m_renderMode = TextRenderFlags.DontRender;
m_isInputParsingRequired = true;
ForceMeshUpdate();
m_renderMode = TextRenderFlags.Render;
}
/// <summary>
/// Function to force the regeneration of the text object.
/// </summary>
/// <param name="flags"> Flags to control which portions of the geometry gets uploaded.</param>
//public virtual void ForceMeshUpdate(TMP_VertexDataUpdateFlags flags) { }
/// <summary>
/// Function to update the geometry of the main and sub text objects.
/// </summary>
/// <param name="mesh"></param>
/// <param name="index"></param>
public virtual void UpdateGeometry(Mesh mesh, int index) { }
/// <summary>
/// Function to push the updated vertex data into the mesh and renderer.
/// </summary>
public virtual void UpdateVertexData(TMP_VertexDataUpdateFlags flags) { }
/// <summary>
/// Function to push the updated vertex data into the mesh and renderer.
/// </summary>
public virtual void UpdateVertexData() { }
/// <summary>
/// Function to push a new set of vertices to the mesh.
/// </summary>
/// <param name="vertices"></param>
public virtual void SetVertices(Vector3[] vertices) { }
/// <summary>
/// Function to be used to force recomputing of character padding when Shader / Material properties have been changed via script.
/// </summary>
public virtual void UpdateMeshPadding() { }
/// <summary>
///
/// </summary>
//public virtual new void UpdateGeometry() { }
/// <summary>
/// Tweens the CanvasRenderer color associated with this Graphic.
/// </summary>
/// <param name="targetColor">Target color.</param>
/// <param name="duration">Tween duration.</param>
/// <param name="ignoreTimeScale">Should ignore Time.scale?</param>
/// <param name="useAlpha">Should also Tween the alpha channel?</param>
public override void CrossFadeColor(Color targetColor, float duration, bool ignoreTimeScale, bool useAlpha)
{
base.CrossFadeColor(targetColor, duration, ignoreTimeScale, useAlpha);
InternalCrossFadeColor(targetColor, duration, ignoreTimeScale, useAlpha);
}
/// <summary>
/// Tweens the alpha of the CanvasRenderer color associated with this Graphic.
/// </summary>
/// <param name="alpha">Target alpha.</param>
/// <param name="duration">Duration of the tween in seconds.</param>
/// <param name="ignoreTimeScale">Should ignore Time.scale?</param>
public override void CrossFadeAlpha(float alpha, float duration, bool ignoreTimeScale)
{
base.CrossFadeAlpha(alpha, duration, ignoreTimeScale);
InternalCrossFadeAlpha(alpha, duration, ignoreTimeScale);
}
/// <summary>
///
/// </summary>
/// <param name="targetColor"></param>
/// <param name="duration"></param>
/// <param name="ignoreTimeScale"></param>
/// <param name="useAlpha"></param>
/// <param name="useRGB"></param>
protected virtual void InternalCrossFadeColor(Color targetColor, float duration, bool ignoreTimeScale, bool useAlpha) { }
/// <summary>
///
/// </summary>
/// <param name="alpha"></param>
/// <param name="duration"></param>
/// <param name="ignoreTimeScale"></param>
protected virtual void InternalCrossFadeAlpha(float alpha, float duration, bool ignoreTimeScale) { }
/// <summary>
/// Method to parse the input text based on its source
/// </summary>
protected void ParseInputText()
{
//Debug.Log("Re-parsing Text.");
////Profiler.BeginSample("ParseInputText()");
m_isInputParsingRequired = false;
switch (m_inputSource)
{
case TextInputSources.String:
case TextInputSources.Text:
StringToCharArray(m_text, ref m_char_buffer);
break;
case TextInputSources.SetText:
SetTextArrayToCharArray(m_input_CharArray, ref m_char_buffer);
break;
case TextInputSources.SetCharArray:
break;
}
SetArraySizes(m_char_buffer);
////Profiler.EndSample();
}
/// <summary>
///
/// </summary>
/// <param name="text"></param>
public void SetText(string text)
{
SetText(text, true);
}
/// <summary>
///
/// </summary>
/// <param name="text"></param>
public void SetText(string text, bool syncTextInputBox)
{
//if (text == old_text) return;
//old_text = text;
m_inputSource = TextInputSources.SetCharArray;
StringToCharArray(text, ref m_char_buffer);
#if UNITY_EDITOR
// Set the text in the Text Input Box in the Unity Editor only.
// TODO: Could revise to convert to string literal
if (syncTextInputBox)
m_text = text;
#endif
m_isInputParsingRequired = true;
m_havePropertiesChanged = true;
m_isCalculateSizeRequired = true;
SetVerticesDirty();
SetLayoutDirty();
}
/// <summary>
/// <para>Formatted string containing a pattern and a value representing the text to be rendered.</para>
/// <para>ex. TextMeshPro.SetText ("Number is {0:1}.", 5.56f);</para>
/// </summary>
/// <typeparam name="T"></typeparam>
/// <param name="text">String containing the pattern."</param>
/// <param name="arg0">Value is a float.</param>
public void SetText(string text, float arg0)
{
SetText(text, arg0, 255, 255);
}
/// <summary>
/// <para>Formatted string containing a pattern and a value representing the text to be rendered.</para>
/// <para>ex. TextMeshPro.SetText ("First number is {0} and second is {1:2}.", 10, 5.756f);</para>
/// </summary>
/// <typeparam name="T"></typeparam>
/// <param name="text">String containing the pattern."</param>
/// <param name="arg0">Value is a float.</param>
/// <param name="arg1">Value is a float.</param>
public void SetText(string text, float arg0, float arg1)
{
SetText(text, arg0, arg1, 255);
}
/// <summary>
/// <para>Formatted string containing a pattern and a value representing the text to be rendered.</para>
/// <para>ex. TextMeshPro.SetText ("A = {0}, B = {1} and C = {2}.", 2, 5, 7);</para>
/// </summary>
/// <typeparam name="T"></typeparam>
/// <param name="text">String containing the pattern."</param>
/// <param name="arg0">Value is a float.</param>
/// <param name="arg1">Value is a float.</param>
/// <param name="arg2">Value is a float.</param>
public void SetText(string text, float arg0, float arg1, float arg2)
{
// Early out if nothing has been changed from previous invocation.
//if (text == old_text && arg0 == old_arg0 && arg1 == old_arg1 && arg2 == old_arg2)
//{
// return;
//}
//old_text = text;
//old_arg1 = 255;
//old_arg2 = 255;
int decimalPrecision = 0;
int index = 0;
for (int i = 0; i < text.Length; i++)
{
char c = text[i];
if (c == 123) // '{'
{
// Check if user is requesting some decimal precision. Format is {0:2}
if (text[i + 2] == 58) // ':'
{
decimalPrecision = text[i + 3] - 48;
}
switch (text[i + 1] - 48)
{
case 0: // 1st Arg
//old_arg0 = arg0;
AddFloatToCharArray(arg0, ref index, decimalPrecision);
break;
case 1: // 2nd Arg
//old_arg1 = arg1;
AddFloatToCharArray(arg1, ref index, decimalPrecision);
break;
case 2: // 3rd Arg
//old_arg2 = arg2;
AddFloatToCharArray(arg2, ref index, decimalPrecision);
break;
}
if (text[i + 2] == 58)
i += 4;
else
i += 2;
continue;
}
m_input_CharArray[index] = c;
index += 1;
}
m_input_CharArray[index] = (char)0;
m_charArray_Length = index; // Set the length to where this '0' termination is.
#if UNITY_EDITOR
// Create new string to be displayed in the Input Text Box of the Editor Panel.
m_text = new string(m_input_CharArray, 0, index);
#endif
m_inputSource = TextInputSources.SetText;
m_isInputParsingRequired = true;
m_havePropertiesChanged = true;
m_isCalculateSizeRequired = true;
SetVerticesDirty();
SetLayoutDirty();
}
/// <summary>
/// Set the text using a StringBuilder.
/// </summary>
/// <description>
/// Using a StringBuilder instead of concatenating strings prevents memory pollution with temporary objects.
/// </description>
/// <param name="text">StringBuilder with text to display.</param>
public void SetText(StringBuilder text)
{
m_inputSource = TextInputSources.SetCharArray;
#if UNITY_EDITOR
// Set the text in the Text Input Box in the Unity Editor only.
m_text = text.ToString();
#endif
StringBuilderToIntArray(text, ref m_char_buffer);
m_isInputParsingRequired = true;
m_havePropertiesChanged = true;
m_isCalculateSizeRequired = true;
SetVerticesDirty();
SetLayoutDirty();
}
/// <summary>
/// Character array containing the text to be displayed.
/// </summary>
/// <param name="sourceText"></param>
public void SetCharArray(char[] sourceText)
{
// Initialize internal character buffer if necessary
if (m_char_buffer == null) m_char_buffer = new int[8];
#if UNITY_EDITOR
// Create new string to be displayed in the Input Text Box of the Editor Panel.
if (sourceText == null || sourceText.Length == 0)
m_text = string.Empty;
else
m_text = new string(sourceText);
#endif
// Clear the Style stack.
m_styleStack.Clear();
int writeIndex = 0;
for (int i = 0; sourceText != null && i < sourceText.Length; i++)
{
if (sourceText[i] == 92 && i < sourceText.Length - 1)
{
switch ((int)sourceText[i + 1])
{
case 110: // \n LineFeed
if (writeIndex == m_char_buffer.Length) ResizeInternalArray(ref m_char_buffer);
m_char_buffer[writeIndex] = (char)10;
i += 1;
writeIndex += 1;
continue;
case 114: // \r LineFeed
if (writeIndex == m_char_buffer.Length) ResizeInternalArray(ref m_char_buffer);
m_char_buffer[writeIndex] = (char)13;
i += 1;
writeIndex += 1;
continue;
case 116: // \t Tab
if (writeIndex == m_char_buffer.Length) ResizeInternalArray(ref m_char_buffer);
m_char_buffer[writeIndex] = (char)9;
i += 1;
writeIndex += 1;
continue;
}
}
// Handle inline replacement of <stlye> and <br> tags.
if (sourceText[i] == 60)
{
if (IsTagName(ref sourceText, "<BR>", i))
{
if (writeIndex == m_char_buffer.Length) ResizeInternalArray(ref m_char_buffer);
m_char_buffer[writeIndex] = 10; ;
writeIndex += 1;
i += 3;
continue;
}
else if (IsTagName(ref sourceText, "<STYLE=", i))
{
int srcOffset = 0;
if (ReplaceOpeningStyleTag(ref sourceText, i, out srcOffset, ref m_char_buffer, ref writeIndex))
{
i = srcOffset;
continue;
}
}
else if (IsTagName(ref sourceText, "</STYLE>", i))
{
ReplaceClosingStyleTag(ref sourceText, i, ref m_char_buffer, ref writeIndex);
// Strip </style> even if style is invalid.
i += 7;
continue;
}
}
if (writeIndex == m_char_buffer.Length) ResizeInternalArray(ref m_char_buffer);
m_char_buffer[writeIndex] = sourceText[i];
writeIndex += 1;
}
if (writeIndex == m_char_buffer.Length) ResizeInternalArray(ref m_char_buffer);
m_char_buffer[writeIndex] = (char)0;
m_inputSource = TextInputSources.SetCharArray;
m_isInputParsingRequired = true;
m_havePropertiesChanged = true;
m_isCalculateSizeRequired = true;
SetVerticesDirty();
SetLayoutDirty();
}
/// <summary>
/// Character array containing the text to be displayed.
/// </summary>
/// <param name="sourceText"></param>
public void SetCharArray(char[] sourceText, int start, int length)
{
// Initialize internal character buffer if necessary
if (m_char_buffer == null) m_char_buffer = new int[8];
#if UNITY_EDITOR
// Create new string to be displayed in the Input Text Box of the Editor Panel.
if (sourceText == null || sourceText.Length == 0 || length == 0)
{
m_text = string.Empty;
start = 0;
length = 0;
}
else
{
// TODO: Add potential range check on start + length relative to array size.
m_text = new string(sourceText, start, length);
}
#endif
// Clear the Style stack.
m_styleStack.Clear();
int writeIndex = 0;
int i = start;
int end = start + length;
for (; i < end; i++)
{
if (sourceText[i] == 92 && i < length - 1)
{
switch ((int)sourceText[i + 1])
{
case 110: // \n LineFeed
if (writeIndex == m_char_buffer.Length) ResizeInternalArray(ref m_char_buffer);
m_char_buffer[writeIndex] = (char)10;
i += 1;
writeIndex += 1;
continue;
case 114: // \r LineFeed
if (writeIndex == m_char_buffer.Length) ResizeInternalArray(ref m_char_buffer);
m_char_buffer[writeIndex] = (char)13;
i += 1;
writeIndex += 1;
continue;
case 116: // \t Tab
if (writeIndex == m_char_buffer.Length) ResizeInternalArray(ref m_char_buffer);
m_char_buffer[writeIndex] = (char)9;
i += 1;
writeIndex += 1;
continue;
}
}
// Handle inline replacement of <stlye> and <br> tags.
if (sourceText[i] == 60)
{
if (IsTagName(ref sourceText, "<BR>", i))
{
if (writeIndex == m_char_buffer.Length) ResizeInternalArray(ref m_char_buffer);
m_char_buffer[writeIndex] = 10; ;
writeIndex += 1;
i += 3;
continue;
}
else if (IsTagName(ref sourceText, "<STYLE=", i))
{
int srcOffset = 0;
if (ReplaceOpeningStyleTag(ref sourceText, i, out srcOffset, ref m_char_buffer, ref writeIndex))
{
i = srcOffset;
continue;
}
}
else if (IsTagName(ref sourceText, "</STYLE>", i))
{
ReplaceClosingStyleTag(ref sourceText, i, ref m_char_buffer, ref writeIndex);
// Strip </style> even if style is invalid.
i += 7;
continue;
}
}
if (writeIndex == m_char_buffer.Length) ResizeInternalArray(ref m_char_buffer);
m_char_buffer[writeIndex] = sourceText[i];
writeIndex += 1;
}
if (writeIndex == m_char_buffer.Length) ResizeInternalArray(ref m_char_buffer);
m_char_buffer[writeIndex] = (char)0;
m_inputSource = TextInputSources.SetCharArray;
m_havePropertiesChanged = true;
m_isInputParsingRequired = true;
m_isCalculateSizeRequired = true;
SetVerticesDirty();
SetLayoutDirty();
}
/// <summary>
/// Character array containing the text to be displayed.
/// </summary>
/// <param name="sourceText"></param>
public void SetCharArray(int[] sourceText, int start, int length)
{
// Initialize internal character buffer if necessary
if (m_char_buffer == null) m_char_buffer = new int[8];
#if UNITY_EDITOR
// Create new string to be displayed in the Input Text Box of the Editor Panel.
if (sourceText == null || sourceText.Length == 0 || length == 0)
{
m_text = string.Empty;
start = 0;
length = 0;
}
else
{
m_text = sourceText.IntToString(start, length);
}
#endif
// Clear the Style stack.
m_styleStack.Clear();
int writeIndex = 0;
int end = start + length;
for (int i = start; i < end; i++)
{
if (sourceText[i] == 92 && i < length - 1)
{
switch ((int)sourceText[i + 1])
{
case 110: // \n LineFeed
if (writeIndex == m_char_buffer.Length) ResizeInternalArray(ref m_char_buffer);
m_char_buffer[writeIndex] = (char)10;
i += 1;
writeIndex += 1;
continue;
case 114: // \r LineFeed
if (writeIndex == m_char_buffer.Length) ResizeInternalArray(ref m_char_buffer);
m_char_buffer[writeIndex] = (char)13;
i += 1;
writeIndex += 1;
continue;
case 116: // \t Tab
if (writeIndex == m_char_buffer.Length) ResizeInternalArray(ref m_char_buffer);
m_char_buffer[writeIndex] = (char)9;
i += 1;
writeIndex += 1;
continue;
}
}
// Handle inline replacement of <stlye> and <br> tags.
if (sourceText[i] == 60)
{
if (IsTagName(ref sourceText, "<BR>", i))
{
if (writeIndex == m_char_buffer.Length) ResizeInternalArray(ref m_char_buffer);
m_char_buffer[writeIndex] = 10; ;
writeIndex += 1;
i += 3;
continue;
}
else if (IsTagName(ref sourceText, "<STYLE=", i))
{
int srcOffset = 0;
if (ReplaceOpeningStyleTag(ref sourceText, i, out srcOffset, ref m_char_buffer, ref writeIndex))
{
i = srcOffset;
continue;
}
}
else if (IsTagName(ref sourceText, "</STYLE>", i))
{
ReplaceClosingStyleTag(ref sourceText, i, ref m_char_buffer, ref writeIndex);
// Strip </style> even if style is invalid.
i += 7;
continue;
}
}
if (writeIndex == m_char_buffer.Length) ResizeInternalArray(ref m_char_buffer);
m_char_buffer[writeIndex] = sourceText[i];
writeIndex += 1;
}
if (writeIndex == m_char_buffer.Length) ResizeInternalArray(ref m_char_buffer);
m_char_buffer[writeIndex] = (char)0;
m_inputSource = TextInputSources.SetCharArray;
m_havePropertiesChanged = true;
m_isInputParsingRequired = true;
m_isCalculateSizeRequired = true;
SetVerticesDirty();
SetLayoutDirty();
}
/// <summary>
/// Copies Content of formatted SetText() to charBuffer.
/// </summary>
/// <param name="sourceText"></param>
/// <param name="charBuffer"></param>
protected void SetTextArrayToCharArray(char[] sourceText, ref int[] charBuffer)
{
//Debug.Log("SetText Array to Char called.");
if (sourceText == null || m_charArray_Length == 0)
return;
if (charBuffer == null) charBuffer = new int[8];
// Clear the Style stack.
m_styleStack.Clear();
int writeIndex = 0;
for (int i = 0; i < m_charArray_Length; i++)
{
// Handle UTF-32 in the input text (string).
if (char.IsHighSurrogate(sourceText[i]) && char.IsLowSurrogate(sourceText[i + 1]))
{
if (writeIndex == charBuffer.Length) ResizeInternalArray(ref charBuffer);
charBuffer[writeIndex] = char.ConvertToUtf32(sourceText[i], sourceText[i + 1]);
i += 1;
writeIndex += 1;
continue;
}
// Handle inline replacement of <stlye> and <br> tags.
if (sourceText[i] == 60)
{
if (IsTagName(ref sourceText, "<BR>", i))
{
if (writeIndex == charBuffer.Length) ResizeInternalArray(ref charBuffer);
charBuffer[writeIndex] = 10;
writeIndex += 1;
i += 3;
continue;
}
else if (IsTagName(ref sourceText, "<STYLE=", i))
{
int srcOffset = 0;
if (ReplaceOpeningStyleTag(ref sourceText, i, out srcOffset, ref charBuffer, ref writeIndex))
{
i = srcOffset;
continue;
}
}
else if (IsTagName(ref sourceText, "</STYLE>", i))
{
ReplaceClosingStyleTag(ref sourceText, i, ref charBuffer, ref writeIndex);
// Strip </style> even if style is invalid.
i += 7;
continue;
}
}
if (writeIndex == charBuffer.Length) ResizeInternalArray(ref charBuffer);
charBuffer[writeIndex] = sourceText[i];
writeIndex += 1;
}
if (writeIndex == charBuffer.Length) ResizeInternalArray(ref charBuffer);
charBuffer[writeIndex] = 0;
}
/// <summary>
/// Method to store the content of a string into an integer array.
/// </summary>
/// <param name="sourceText"></param>
/// <param name="charBuffer"></param>
protected void StringToCharArray(string sourceText, ref int[] charBuffer)
{
if (sourceText == null)
{
charBuffer[0] = 0;
return;
}
if (charBuffer == null) charBuffer = new int[8];
// Clear the Style stack.
m_styleStack.SetDefault(0);
int writeIndex = 0;
for (int i = 0; i < sourceText.Length; i++)
{
if (m_inputSource == TextInputSources.Text && sourceText[i] == 92 && sourceText.Length > i + 1)
{
switch ((int)sourceText[i + 1])
{
case 85: // \U00000000 for UTF-32 Unicode
if (sourceText.Length > i + 9)
{
if (writeIndex == charBuffer.Length) ResizeInternalArray(ref charBuffer);
charBuffer[writeIndex] = GetUTF32(sourceText, i + 2);
i += 9;
writeIndex += 1;
continue;
}
break;
case 92: // \ escape
if (!m_parseCtrlCharacters) break;
if (sourceText.Length <= i + 2) break;
if (writeIndex + 2 > charBuffer.Length) ResizeInternalArray(ref charBuffer);
charBuffer[writeIndex] = sourceText[i + 1];
charBuffer[writeIndex + 1] = sourceText[i + 2];
i += 2;
writeIndex += 2;
continue;
case 110: // \n LineFeed
if (!m_parseCtrlCharacters) break;
if (writeIndex == charBuffer.Length) ResizeInternalArray(ref charBuffer);
charBuffer[writeIndex] = (char)10;
i += 1;
writeIndex += 1;
continue;
case 114: // \r
if (!m_parseCtrlCharacters) break;
if (writeIndex == charBuffer.Length) ResizeInternalArray(ref charBuffer);
charBuffer[writeIndex] = (char)13;
i += 1;
writeIndex += 1;
continue;
case 116: // \t Tab
if (!m_parseCtrlCharacters) break;
if (writeIndex == charBuffer.Length) ResizeInternalArray(ref charBuffer);
charBuffer[writeIndex] = (char)9;
i += 1;
writeIndex += 1;
continue;
case 117: // \u0000 for UTF-16 Unicode
if (sourceText.Length > i + 5)
{
if (writeIndex == charBuffer.Length) ResizeInternalArray(ref charBuffer);
charBuffer[writeIndex] = (char)GetUTF16(sourceText, i + 2);
i += 5;
writeIndex += 1;
continue;
}
break;
}
}
// Handle UTF-32 in the input text (string). // Not sure this is needed //
if (char.IsHighSurrogate(sourceText[i]) && char.IsLowSurrogate(sourceText[i + 1]))
{
if (writeIndex == charBuffer.Length) ResizeInternalArray(ref charBuffer);
charBuffer[writeIndex] = char.ConvertToUtf32(sourceText[i], sourceText[i + 1]);
i += 1;
writeIndex += 1;
continue;
}
//// Handle inline replacement of <stlye> and <br> tags.
if (sourceText[i] == 60 && m_isRichText)
{
if (IsTagName(ref sourceText, "<BR>", i))
{
if (writeIndex == charBuffer.Length) ResizeInternalArray(ref charBuffer);
charBuffer[writeIndex] = 10; ;
writeIndex += 1;
i += 3;
continue;
}
else if (IsTagName(ref sourceText, "<STYLE=", i))
{
int srcOffset = 0;
if (ReplaceOpeningStyleTag(ref sourceText, i, out srcOffset, ref charBuffer, ref writeIndex))
{
i = srcOffset;
continue;
}
}
else if (IsTagName(ref sourceText, "</STYLE>", i))
{
ReplaceClosingStyleTag(ref sourceText, i, ref charBuffer, ref writeIndex);
// Strip </style> even if style is invalid.
i += 7;
continue;
}
}
if (writeIndex == charBuffer.Length) ResizeInternalArray(ref charBuffer);
charBuffer[writeIndex] = sourceText[i];
writeIndex += 1;
}
if (writeIndex == charBuffer.Length) ResizeInternalArray(ref charBuffer);
charBuffer[writeIndex] = (char)0;
}
/// <summary>
/// Copy contents of StringBuilder into int array.
/// </summary>
/// <param name="sourceText">Text to copy.</param>
/// <param name="charBuffer">Array to store contents.</param>
protected void StringBuilderToIntArray(StringBuilder sourceText, ref int[] charBuffer)
{
if (sourceText == null)
{
charBuffer[0] = 0;
return;
}
if (charBuffer == null) charBuffer = new int[8];
// Clear the Style stack.
m_styleStack.Clear();
#if UNITY_EDITOR
// Create new string to be displayed in the Input Text Box of the Editor Panel.
m_text = sourceText.ToString();
#endif
int writeIndex = 0;
for (int i = 0; i < sourceText.Length; i++)
{
if (m_parseCtrlCharacters && sourceText[i] == 92 && sourceText.Length > i + 1)
{
switch ((int)sourceText[i + 1])
{
case 85: // \U00000000 for UTF-32 Unicode
if (sourceText.Length > i + 9)
{
if (writeIndex == charBuffer.Length) ResizeInternalArray(ref charBuffer);
charBuffer[writeIndex] = GetUTF32(sourceText, i + 2);
i += 9;
writeIndex += 1;
continue;
}
break;
case 92: // \ escape
if (sourceText.Length <= i + 2) break;
if (writeIndex + 2 > charBuffer.Length) ResizeInternalArray(ref charBuffer);
charBuffer[writeIndex] = sourceText[i + 1];
charBuffer[writeIndex + 1] = sourceText[i + 2];
i += 2;
writeIndex += 2;
continue;
case 110: // \n LineFeed
if (writeIndex == charBuffer.Length) ResizeInternalArray(ref charBuffer);
charBuffer[writeIndex] = (char)10;
i += 1;
writeIndex += 1;
continue;
case 114: // \r
if (writeIndex == charBuffer.Length) ResizeInternalArray(ref charBuffer);
charBuffer[writeIndex] = (char)13;
i += 1;
writeIndex += 1;
continue;
case 116: // \t Tab
if (writeIndex == charBuffer.Length) ResizeInternalArray(ref charBuffer);
charBuffer[writeIndex] = (char)9;
i += 1;
writeIndex += 1;
continue;
case 117: // \u0000 for UTF-16 Unicode
if (sourceText.Length > i + 5)
{
if (writeIndex == charBuffer.Length) ResizeInternalArray(ref charBuffer);
charBuffer[writeIndex] = (char)GetUTF16(sourceText, i + 2);
i += 5;
writeIndex += 1;
continue;
}
break;
}
}
// Handle UTF-32 in the input text (string).
if (char.IsHighSurrogate(sourceText[i]) && char.IsLowSurrogate(sourceText[i + 1]))
{
if (writeIndex == charBuffer.Length) ResizeInternalArray(ref charBuffer);
charBuffer[writeIndex] = char.ConvertToUtf32(sourceText[i], sourceText[i + 1]);
i += 1;
writeIndex += 1;
continue;
}
// Handle inline replacement of <stlye> and <br> tags.
if (sourceText[i] == 60)
{
if (IsTagName(ref sourceText, "<BR>", i))
{
if (writeIndex == charBuffer.Length) ResizeInternalArray(ref charBuffer);
charBuffer[writeIndex] = 10;
writeIndex += 1;
i += 3;
continue;
}
else if (IsTagName(ref sourceText, "<STYLE=", i))
{
int srcOffset = 0;
if (ReplaceOpeningStyleTag(ref sourceText, i, out srcOffset, ref charBuffer, ref writeIndex))
{
i = srcOffset;
continue;
}
}
else if (IsTagName(ref sourceText, "</STYLE>", i))
{
ReplaceClosingStyleTag(ref sourceText, i, ref charBuffer, ref writeIndex);
// Strip </style> even if style is invalid.
i += 7;
continue;
}
}
if (writeIndex == charBuffer.Length) ResizeInternalArray(ref charBuffer);
charBuffer[writeIndex] = sourceText[i];
writeIndex += 1;
}
if (writeIndex == charBuffer.Length) ResizeInternalArray(ref charBuffer);
charBuffer[writeIndex] = (char)0;
}
/// <summary>
/// Method to handle inline replacement of style tag by opening style definition.
/// </summary>
/// <param name="sourceText"></param>
/// <param name="srcIndex"></param>
/// <param name="srcOffset"></param>
/// <param name="charBuffer"></param>
/// <param name="writeIndex"></param>
/// <returns></returns>
bool ReplaceOpeningStyleTag(ref string sourceText, int srcIndex, out int srcOffset, ref int[] charBuffer, ref int writeIndex)
{
// Validate <style> tag.
int hashCode = GetTagHashCode(ref sourceText, srcIndex + 7, out srcOffset);
TMP_Style style = TMP_StyleSheet.GetStyle(hashCode);
// Return if we don't have a valid style.
if (style == null || srcOffset == 0) return false;
m_styleStack.Add(style.hashCode);
int styleLength = style.styleOpeningTagArray.Length;
// Replace <style> tag with opening definition
int[] openingTagArray = style.styleOpeningTagArray;
for (int i = 0; i < styleLength; i++)
{
int c = openingTagArray[i];
if (c == 60)
{
if (IsTagName(ref openingTagArray, "<BR>", i))
{
if (writeIndex == charBuffer.Length) ResizeInternalArray(ref charBuffer);
charBuffer[writeIndex] = 10;
writeIndex += 1;
i += 3;
continue;
}
else if (IsTagName(ref openingTagArray, "<STYLE=", i))
{
int offset = 0;
if (ReplaceOpeningStyleTag(ref openingTagArray, i, out offset, ref charBuffer, ref writeIndex))
{
i = offset;
continue;
}
}
else if (IsTagName(ref openingTagArray, "</STYLE>", i))
{
ReplaceClosingStyleTag(ref openingTagArray, i, ref charBuffer, ref writeIndex);
// Strip </style> even if style is invalid.
i += 7;
continue;
}
}
if (writeIndex == charBuffer.Length) ResizeInternalArray(ref charBuffer);
charBuffer[writeIndex] = c;
writeIndex += 1;
}
return true;
}
/// <summary>
/// Method to handle inline replacement of style tag by opening style definition.
/// </summary>
/// <param name="sourceText"></param>
/// <param name="srcIndex"></param>
/// <param name="srcOffset"></param>
/// <param name="charBuffer"></param>
/// <param name="writeIndex"></param>
/// <returns></returns>
bool ReplaceOpeningStyleTag(ref int[] sourceText, int srcIndex, out int srcOffset, ref int[] charBuffer, ref int writeIndex)
{
// Validate <style> tag.
int hashCode = GetTagHashCode(ref sourceText, srcIndex + 7, out srcOffset);
TMP_Style style = TMP_StyleSheet.GetStyle(hashCode);
// Return if we don't have a valid style.
if (style == null || srcOffset == 0) return false;
m_styleStack.Add(style.hashCode);
int styleLength = style.styleOpeningTagArray.Length;
// Replace <style> tag with opening definition
int[] openingTagArray = style.styleOpeningTagArray;
for (int i = 0; i < styleLength; i++)
{
int c = openingTagArray[i];
if (c == 60)
{
if (IsTagName(ref openingTagArray, "<BR>", i))
{
if (writeIndex == charBuffer.Length) ResizeInternalArray(ref charBuffer);
charBuffer[writeIndex] = 10;
writeIndex += 1;
i += 3;
continue;
}
else if (IsTagName(ref openingTagArray, "<STYLE=", i))
{
int offset = 0;
if (ReplaceOpeningStyleTag(ref openingTagArray, i, out offset, ref charBuffer, ref writeIndex))
{
i = offset;
continue;
}
}
else if (IsTagName(ref openingTagArray, "</STYLE>", i))
{
ReplaceClosingStyleTag(ref openingTagArray, i, ref charBuffer, ref writeIndex);
// Strip </style> even if style is invalid.
i += 7;
continue;
}
}
if (writeIndex == charBuffer.Length) ResizeInternalArray(ref charBuffer);
charBuffer[writeIndex] = c;
writeIndex += 1;
}
return true;
}
/// <summary>
/// Method to handle inline replacement of style tag by opening style definition.
/// </summary>
/// <param name="sourceText"></param>
/// <param name="srcIndex"></param>
/// <param name="srcOffset"></param>
/// <param name="charBuffer"></param>
/// <param name="writeIndex"></param>
/// <returns></returns>
bool ReplaceOpeningStyleTag(ref char[] sourceText, int srcIndex, out int srcOffset, ref int[] charBuffer, ref int writeIndex)
{
// Validate <style> tag.
int hashCode = GetTagHashCode(ref sourceText, srcIndex + 7, out srcOffset);
TMP_Style style = TMP_StyleSheet.GetStyle(hashCode);
// Return if we don't have a valid style.
if (style == null || srcOffset == 0) return false;
m_styleStack.Add(style.hashCode);
int styleLength = style.styleOpeningTagArray.Length;
// Replace <style> tag with opening definition
int[] openingTagArray = style.styleOpeningTagArray;
for (int i = 0; i < styleLength; i++)
{
int c = openingTagArray[i];
if (c == 60)
{
if (IsTagName(ref openingTagArray, "<BR>", i))
{
if (writeIndex == charBuffer.Length) ResizeInternalArray(ref charBuffer);
charBuffer[writeIndex] = 10;
writeIndex += 1;
i += 3;
continue;
}
else if (IsTagName(ref openingTagArray, "<STYLE=", i))
{
int offset = 0;
if (ReplaceOpeningStyleTag(ref openingTagArray, i, out offset, ref charBuffer, ref writeIndex))
{
i = offset;
continue;
}
}
else if (IsTagName(ref openingTagArray, "</STYLE>", i))
{
ReplaceClosingStyleTag(ref openingTagArray, i, ref charBuffer, ref writeIndex);
// Strip </style> even if style is invalid.
i += 7;
continue;
}
}
if (writeIndex == charBuffer.Length) ResizeInternalArray(ref charBuffer);
charBuffer[writeIndex] = c;
writeIndex += 1;
}
return true;
}
/// <summary>
/// Method to handle inline replacement of style tag by opening style definition.
/// </summary>
/// <param name="sourceText"></param>
/// <param name="srcIndex"></param>
/// <param name="srcOffset"></param>
/// <param name="charBuffer"></param>
/// <param name="writeIndex"></param>
/// <returns></returns>
bool ReplaceOpeningStyleTag(ref StringBuilder sourceText, int srcIndex, out int srcOffset, ref int[] charBuffer, ref int writeIndex)
{
// Validate <style> tag.
int hashCode = GetTagHashCode(ref sourceText, srcIndex + 7, out srcOffset);
TMP_Style style = TMP_StyleSheet.GetStyle(hashCode);
// Return if we don't have a valid style.
if (style == null || srcOffset == 0) return false;
m_styleStack.Add(style.hashCode);
int styleLength = style.styleOpeningTagArray.Length;
// Replace <style> tag with opening definition
int[] openingTagArray = style.styleOpeningTagArray;
for (int i = 0; i < styleLength; i++)
{
int c = openingTagArray[i];
if (c == 60)
{
if (IsTagName(ref openingTagArray, "<BR>", i))
{
if (writeIndex == charBuffer.Length) ResizeInternalArray(ref charBuffer);
charBuffer[writeIndex] = 10;
writeIndex += 1;
i += 3;
continue;
}
else if (IsTagName(ref openingTagArray, "<STYLE=", i))
{
int offset = 0;
if (ReplaceOpeningStyleTag(ref openingTagArray, i, out offset, ref charBuffer, ref writeIndex))
{
i = offset;
continue;
}
}
else if (IsTagName(ref openingTagArray, "</STYLE>", i))
{
ReplaceClosingStyleTag(ref openingTagArray, i, ref charBuffer, ref writeIndex);
// Strip </style> even if style is invalid.
i += 7;
continue;
}
}
if (writeIndex == charBuffer.Length) ResizeInternalArray(ref charBuffer);
charBuffer[writeIndex] = c;
writeIndex += 1;
}
return true;
}
/// <summary>
/// Method to handle inline replacement of style tag by closing style definition.
/// </summary>
/// <param name="sourceText"></param>
/// <param name="srcIndex"></param>
/// <param name="charBuffer"></param>
/// <param name="writeIndex"></param>
/// <returns></returns>
bool ReplaceClosingStyleTag(ref string sourceText, int srcIndex, ref int[] charBuffer, ref int writeIndex)
{
// Get style from the Style Stack
int hashCode = m_styleStack.CurrentItem();
TMP_Style style = TMP_StyleSheet.GetStyle(hashCode);
m_styleStack.Remove();
// Return if we don't have a valid style.
if (style == null) return false;
int styleLength = style.styleClosingTagArray.Length;
// Replace <style> tag with opening definition
int[] closingTagArray = style.styleClosingTagArray;
for (int i = 0; i < styleLength; i++)
{
int c = closingTagArray[i];
if (c == 60)
{
if (IsTagName(ref closingTagArray, "<BR>", i))
{
if (writeIndex == charBuffer.Length) ResizeInternalArray(ref charBuffer);
charBuffer[writeIndex] = 10;
writeIndex += 1;
i += 3;
continue;
}
else if (IsTagName(ref closingTagArray, "<STYLE=", i))
{
int offset = 0;
if (ReplaceOpeningStyleTag(ref closingTagArray, i, out offset, ref charBuffer, ref writeIndex))
{
i = offset;
continue;
}
}
else if (IsTagName(ref closingTagArray, "</STYLE>", i))
{
ReplaceClosingStyleTag(ref closingTagArray, i, ref charBuffer, ref writeIndex);
// Strip </style> even if style is invalid.
i += 7;
continue;
}
}
if (writeIndex == charBuffer.Length) ResizeInternalArray(ref charBuffer);
charBuffer[writeIndex] = c;
writeIndex += 1;
}
return true;
}
/// <summary>
/// Method to handle inline replacement of style tag by closing style definition.
/// </summary>
/// <param name="sourceText"></param>
/// <param name="srcIndex"></param>
/// <param name="charBuffer"></param>
/// <param name="writeIndex"></param>
/// <returns></returns>
bool ReplaceClosingStyleTag(ref int[] sourceText, int srcIndex, ref int[] charBuffer, ref int writeIndex)
{
// Get style from the Style Stack
int hashCode = m_styleStack.CurrentItem();
TMP_Style style = TMP_StyleSheet.GetStyle(hashCode);
m_styleStack.Remove();
// Return if we don't have a valid style.
if (style == null) return false;
int styleLength = style.styleClosingTagArray.Length;
// Replace <style> tag with opening definition
int[] closingTagArray = style.styleClosingTagArray;
for (int i = 0; i < styleLength; i++)
{
int c = closingTagArray[i];
if (c == 60)
{
if (IsTagName(ref closingTagArray, "<BR>", i))
{
if (writeIndex == charBuffer.Length) ResizeInternalArray(ref charBuffer);
charBuffer[writeIndex] = 10;
writeIndex += 1;
i += 3;
continue;
}
else if (IsTagName(ref closingTagArray, "<STYLE=", i))
{
int offset = 0;
if (ReplaceOpeningStyleTag(ref closingTagArray, i, out offset, ref charBuffer, ref writeIndex))
{
i = offset;
continue;
}
}
else if (IsTagName(ref closingTagArray, "</STYLE>", i))
{
ReplaceClosingStyleTag(ref closingTagArray, i, ref charBuffer, ref writeIndex);
// Strip </style> even if style is invalid.
i += 7;
continue;
}
}
if (writeIndex == charBuffer.Length) ResizeInternalArray(ref charBuffer);
charBuffer[writeIndex] = c;
writeIndex += 1;
}
return true;
}
/// <summary>
/// Method to handle inline replacement of style tag by closing style definition.
/// </summary>
/// <param name="sourceText"></param>
/// <param name="srcIndex"></param>
/// <param name="charBuffer"></param>
/// <param name="writeIndex"></param>
/// <returns></returns>
bool ReplaceClosingStyleTag(ref char[] sourceText, int srcIndex, ref int[] charBuffer, ref int writeIndex)
{
// Get style from the Style Stack
int hashCode = m_styleStack.CurrentItem();
TMP_Style style = TMP_StyleSheet.GetStyle(hashCode);
m_styleStack.Remove();
// Return if we don't have a valid style.
if (style == null) return false;
int styleLength = style.styleClosingTagArray.Length;
// Replace <style> tag with opening definition
int[] closingTagArray = style.styleClosingTagArray;
for (int i = 0; i < styleLength; i++)
{
int c = closingTagArray[i];
if (c == 60)
{
if (IsTagName(ref closingTagArray, "<BR>", i))
{
if (writeIndex == charBuffer.Length) ResizeInternalArray(ref charBuffer);
charBuffer[writeIndex] = 10;
writeIndex += 1;
i += 3;
continue;
}
else if (IsTagName(ref closingTagArray, "<STYLE=", i))
{
int offset = 0;
if (ReplaceOpeningStyleTag(ref closingTagArray, i, out offset, ref charBuffer, ref writeIndex))
{
i = offset;
continue;
}
}
else if (IsTagName(ref closingTagArray, "</STYLE>", i))
{
ReplaceClosingStyleTag(ref closingTagArray, i, ref charBuffer, ref writeIndex);
// Strip </style> even if style is invalid.
i += 7;
continue;
}
}
if (writeIndex == charBuffer.Length) ResizeInternalArray(ref charBuffer);
charBuffer[writeIndex] = c;
writeIndex += 1;
}
return true;
}
/// <summary>
/// Method to handle inline replacement of style tag by closing style definition.
/// </summary>
/// <param name="sourceText"></param>
/// <param name="srcIndex"></param>
/// <param name="charBuffer"></param>
/// <param name="writeIndex"></param>
/// <returns></returns>
bool ReplaceClosingStyleTag(ref StringBuilder sourceText, int srcIndex, ref int[] charBuffer, ref int writeIndex)
{
// Get style from the Style Stack
int hashCode = m_styleStack.CurrentItem();
TMP_Style style = TMP_StyleSheet.GetStyle(hashCode);
m_styleStack.Remove();
// Return if we don't have a valid style.
if (style == null) return false;
int styleLength = style.styleClosingTagArray.Length;
// Replace <style> tag with opening definition
int[] closingTagArray = style.styleClosingTagArray;
for (int i = 0; i < styleLength; i++)
{
int c = closingTagArray[i];
if (c == 60)
{
if (IsTagName(ref closingTagArray, "<BR>", i))
{
if (writeIndex == charBuffer.Length) ResizeInternalArray(ref charBuffer);
charBuffer[writeIndex] = 10;
writeIndex += 1;
i += 3;
continue;
}
else if (IsTagName(ref closingTagArray, "<STYLE=", i))
{
int offset = 0;
if (ReplaceOpeningStyleTag(ref closingTagArray, i, out offset, ref charBuffer, ref writeIndex))
{
i = offset;
continue;
}
}
else if (IsTagName(ref closingTagArray, "</STYLE>", i))
{
ReplaceClosingStyleTag(ref closingTagArray, i, ref charBuffer, ref writeIndex);
// Strip </style> even if style is invalid.
i += 7;
continue;
}
}
if (writeIndex == charBuffer.Length) ResizeInternalArray(ref charBuffer);
charBuffer[writeIndex] = c;
writeIndex += 1;
}
return true;
}
/// <summary>
/// Method to check for a matching rich text tag.
/// </summary>
/// <param name="text"></param>
/// <param name="tag"></param>
/// <param name="index"></param>
/// <returns></returns>
bool IsTagName (ref string text, string tag, int index)
{
if (text.Length < index + tag.Length) return false;
for (int i = 0; i < tag.Length; i++)
{
if (TMP_TextUtilities.ToUpperFast(text[index + i]) != tag[i]) return false;
}
return true;
}
/// <summary>
/// Method to check for a matching rich text tag.
/// </summary>
/// <param name="text"></param>
/// <param name="tag"></param>
/// <param name="index"></param>
/// <returns></returns>
bool IsTagName(ref char[] text, string tag, int index)
{
if (text.Length < index + tag.Length) return false;
for (int i = 0; i < tag.Length; i++)
{
if (TMP_TextUtilities.ToUpperFast(text[index + i]) != tag[i]) return false;
}
return true;
}
/// <summary>
/// Method to check for a matching rich text tag.
/// </summary>
/// <param name="text"></param>
/// <param name="tag"></param>
/// <param name="index"></param>
/// <returns></returns>
bool IsTagName(ref int[] text, string tag, int index)
{
if (text.Length < index + tag.Length) return false;
for (int i = 0; i < tag.Length; i++)
{
if (TMP_TextUtilities.ToUpperFast((char)text[index + i]) != tag[i]) return false;
}
return true;
}
/// <summary>
/// Method to check for a matching rich text tag.
/// </summary>
/// <param name="text"></param>
/// <param name="tag"></param>
/// <param name="index"></param>
/// <returns></returns>
bool IsTagName(ref StringBuilder text, string tag, int index)
{
if (text.Length < index + tag.Length) return false;
for (int i = 0; i < tag.Length; i++)
{
if (TMP_TextUtilities.ToUpperFast(text[index + i]) != tag[i]) return false;
}
return true;
}
/// <summary>
/// Get Hashcode for a given tag.
/// </summary>
/// <param name="text"></param>
/// <param name="index"></param>
/// <param name="closeIndex"></param>
/// <returns></returns>
int GetTagHashCode(ref string text, int index, out int closeIndex)
{
int hashCode = 0;
closeIndex = 0;
for (int i = index; i < text.Length; i++)
{
// Skip quote '"' character
if (text[i] == 34) continue;
// Break at '>'
if (text[i] == 62) { closeIndex = i; break; }
hashCode = (hashCode << 5) + hashCode ^ text[i];
}
return hashCode;
}
/// <summary>
/// Get Hashcode for a given tag.
/// </summary>
/// <param name="text"></param>
/// <param name="index"></param>
/// <param name="closeIndex"></param>
/// <returns></returns>
int GetTagHashCode(ref char[] text, int index, out int closeIndex)
{
int hashCode = 0;
closeIndex = 0;
for (int i = index; i < text.Length; i++)
{
// Skip quote '"' character
if (text[i] == 34) continue;
// Break at '>'
if (text[i] == 62) { closeIndex = i; break; }
hashCode = (hashCode << 5) + hashCode ^ text[i];
}
return hashCode;
}
/// <summary>
/// Get Hashcode for a given tag.
/// </summary>
/// <param name="text"></param>
/// <param name="index"></param>
/// <param name="closeIndex"></param>
/// <returns></returns>
int GetTagHashCode(ref int[] text, int index, out int closeIndex)
{
int hashCode = 0;
closeIndex = 0;
for (int i = index; i < text.Length; i++)
{
// Skip quote '"' character
if (text[i] == 34) continue;
// Break at '>'
if (text[i] == 62) { closeIndex = i; break; }
hashCode = (hashCode << 5) + hashCode ^ text[i];
}
return hashCode;
}
/// <summary>
/// Get Hashcode for a given tag.
/// </summary>
/// <param name="text"></param>
/// <param name="index"></param>
/// <param name="closeIndex"></param>
/// <returns></returns>
int GetTagHashCode(ref StringBuilder text, int index, out int closeIndex)
{
int hashCode = 0;
closeIndex = 0;
for (int i = index; i < text.Length; i++)
{
// Skip quote '"' character
if (text[i] == 34) continue;
// Break at '>'
if (text[i] == 62) { closeIndex = i; break; }
hashCode = (hashCode << 5) + hashCode ^ text[i];
}
return hashCode;
}
/// <summary>
///
/// </summary>
void ResizeInternalArray <T>(ref T[] array)
{
int size = Mathf.NextPowerOfTwo(array.Length + 1);
System.Array.Resize(ref array, size);
}
private readonly float[] k_Power = { 5e-1f, 5e-2f, 5e-3f, 5e-4f, 5e-5f, 5e-6f, 5e-7f, 5e-8f, 5e-9f, 5e-10f }; // Used by FormatText to enable rounding and avoid using Mathf.Pow.
/// <summary>
/// Function used in conjunction with SetText()
/// </summary>
/// <param name="number"></param>
/// <param name="index"></param>
/// <param name="precision"></param>
protected void AddFloatToCharArray(float number, ref int index, int precision)
{
if (number < 0)
{
m_input_CharArray[index++] = '-';
number = -number;
}
number += k_Power[Mathf.Min(9, precision)];
int integer = (int)number;
AddIntToCharArray(integer, ref index, precision);
if (precision > 0)
{
// Add the decimal point
m_input_CharArray[index++] = '.';
number -= integer;
for (int p = 0; p < precision; p++)
{
number *= 10;
int d = (int)(number);
m_input_CharArray[index++] = (char)(d + 48);
number -= d;
}
}
}
/// <summary>
/// // Function used in conjunction with SetText()
/// </summary>
/// <param name="number"></param>
/// <param name="index"></param>
/// <param name="precision"></param>
protected void AddIntToCharArray(int number, ref int index, int precision)
{
if (number < 0)
{
m_input_CharArray[index++] = '-';
number = -number;
}
int i = index;
do
{
m_input_CharArray[i++] = (char)(number % 10 + 48);
number /= 10;
} while (number > 0);
int lastIndex = i;
// Reverse string
while (index + 1 < i)
{
i -= 1;
char t = m_input_CharArray[index];
m_input_CharArray[index] = m_input_CharArray[i];
m_input_CharArray[i] = t;
index += 1;
}
index = lastIndex;
}
/// <summary>
/// Method used to determine the number of visible characters and required buffer allocations.
/// </summary>
/// <param name="chars"></param>
/// <returns></returns>
protected virtual int SetArraySizes(int[] chars) { return 0; }
/// <summary>
/// Method which parses the text input, does the layout of the text as well as generating the geometry.
/// </summary>
protected virtual void GenerateTextMesh() { }
/// <summary>
/// Function to Calculate the Preferred Width and Height of the text object.
/// </summary>
/// <returns></returns>
public Vector2 GetPreferredValues()
{
if (m_isInputParsingRequired || m_isTextTruncated)
{
m_isCalculatingPreferredValues = true;
ParseInputText();
}
// CALCULATE PREFERRED WIDTH
float preferredWidth = GetPreferredWidth();
// CALCULATE PREFERRED HEIGHT
float preferredHeight = GetPreferredHeight();
return new Vector2(preferredWidth, preferredHeight);
}
/// <summary>
/// Function to Calculate the Preferred Width and Height of the text object given the provided width and height.
/// </summary>
/// <returns></returns>
public Vector2 GetPreferredValues(float width, float height)
{
if (m_isInputParsingRequired || m_isTextTruncated)
{
m_isCalculatingPreferredValues = true;
ParseInputText();
}
Vector2 margin = new Vector2(width, height);
// CALCULATE PREFERRED WIDTH
float preferredWidth = GetPreferredWidth(margin);
// CALCULATE PREFERRED HEIGHT
float preferredHeight = GetPreferredHeight(margin);
return new Vector2(preferredWidth, preferredHeight);
}
/// <summary>
/// Function to Calculate the Preferred Width and Height of the text object given a certain string.
/// </summary>
/// <param name="text"></param>
/// <returns></returns>
public Vector2 GetPreferredValues(string text)
{
m_isCalculatingPreferredValues = true;
StringToCharArray(text, ref m_char_buffer);
SetArraySizes(m_char_buffer);
Vector2 margin = k_LargePositiveVector2;
// CALCULATE PREFERRED WIDTH
float preferredWidth = GetPreferredWidth(margin);
// CALCULATE PREFERRED HEIGHT
float preferredHeight = GetPreferredHeight(margin);
return new Vector2(preferredWidth, preferredHeight);
}
/// <summary>
/// Function to Calculate the Preferred Width and Height of the text object given a certain string and size of text container.
/// </summary>
/// <param name="text"></param>
/// <returns></returns>
public Vector2 GetPreferredValues(string text, float width, float height)
{
m_isCalculatingPreferredValues = true;
StringToCharArray(text, ref m_char_buffer);
SetArraySizes(m_char_buffer);
Vector2 margin = new Vector2(width, height);
// CALCULATE PREFERRED WIDTH
float preferredWidth = GetPreferredWidth(margin);
// CALCULATE PREFERRED HEIGHT
float preferredHeight = GetPreferredHeight(margin);
return new Vector2(preferredWidth, preferredHeight);
}
/// <summary>
/// Method to calculate the preferred width of a text object.
/// </summary>
/// <returns></returns>
protected float GetPreferredWidth()
{
if (TMP_Settings.instance == null) return 0;
float fontSize = m_enableAutoSizing ? m_fontSizeMax : m_fontSize;
// Reset auto sizing point size bounds
m_minFontSize = m_fontSizeMin;
m_maxFontSize = m_fontSizeMax;
m_charWidthAdjDelta = 0;
// Set Margins to Infinity
Vector2 margin = k_LargePositiveVector2;
if (m_isInputParsingRequired || m_isTextTruncated)
{
m_isCalculatingPreferredValues = true;
ParseInputText();
}
m_recursiveCount = 0;
float preferredWidth = CalculatePreferredValues(fontSize, margin, true).x;
m_isPreferredWidthDirty = false;
//Debug.Log("GetPreferredWidth() Called at frame " + Time.frameCount + ". Returning width of " + preferredWidth);
return preferredWidth;
}
/// <summary>
/// Method to calculate the preferred width of a text object.
/// </summary>
/// <param name="margin"></param>
/// <returns></returns>
protected float GetPreferredWidth(Vector2 margin)
{
float fontSize = m_enableAutoSizing ? m_fontSizeMax : m_fontSize;
// Reset auto sizing point size bounds
m_minFontSize = m_fontSizeMin;
m_maxFontSize = m_fontSizeMax;
m_charWidthAdjDelta = 0;
m_recursiveCount = 0;
float preferredWidth = CalculatePreferredValues(fontSize, margin, true).x;
//Debug.Log("GetPreferredWidth() Called. Returning width of " + preferredWidth);
return preferredWidth;
}
/// <summary>
/// Method to calculate the preferred height of a text object.
/// </summary>
/// <returns></returns>
protected float GetPreferredHeight()
{
if (TMP_Settings.instance == null) return 0;
float fontSize = m_enableAutoSizing ? m_fontSizeMax : m_fontSize;
// Reset auto sizing point size bounds
m_minFontSize = m_fontSizeMin;
m_maxFontSize = m_fontSizeMax;
m_charWidthAdjDelta = 0;
Vector2 margin = new Vector2(m_marginWidth != 0 ? m_marginWidth : k_LargePositiveFloat, k_LargePositiveFloat);
if (m_isInputParsingRequired || m_isTextTruncated)
{
m_isCalculatingPreferredValues = true;
ParseInputText();
}
m_recursiveCount = 0;
float preferredHeight = CalculatePreferredValues(fontSize, margin, !m_enableAutoSizing).y;
m_isPreferredHeightDirty = false;
//Debug.Log("GetPreferredHeight() Called. Returning height of " + preferredHeight);
return preferredHeight;
}
/// <summary>
/// Method to calculate the preferred height of a text object.
/// </summary>
/// <param name="margin"></param>
/// <returns></returns>
protected float GetPreferredHeight(Vector2 margin)
{
float fontSize = m_enableAutoSizing ? m_fontSizeMax : m_fontSize;
// Reset auto sizing point size bounds
m_minFontSize = m_fontSizeMin;
m_maxFontSize = m_fontSizeMax;
m_charWidthAdjDelta = 0;
m_recursiveCount = 0;
float preferredHeight = CalculatePreferredValues(fontSize, margin, true).y;
//Debug.Log("GetPreferredHeight() Called. Returning height of " + preferredHeight);
return preferredHeight;
}
/// <summary>
/// Method returning the rendered width and height of the text object.
/// </summary>
/// <returns></returns>
public Vector2 GetRenderedValues()
{
return GetTextBounds().size;
}
/// <summary>
///
/// </summary>
/// <param name="onlyVisibleCharacters">Should returned value only factor in visible characters and exclude those greater than maxVisibleCharacters for instance.</param>
/// <returns></returns>
public Vector2 GetRenderedValues(bool onlyVisibleCharacters)
{
return GetTextBounds(onlyVisibleCharacters).size;
}
/// <summary>
/// Method returning the rendered width of the text object.
/// </summary>
/// <returns></returns>
protected float GetRenderedWidth()
{
return GetRenderedValues().x;
}
/// <summary>
/// Method returning the rendered width of the text object.
/// </summary>
/// <returns></returns>
protected float GetRenderedWidth(bool onlyVisibleCharacters)
{
return GetRenderedValues(onlyVisibleCharacters).x;
}
/// <summary>
/// Method returning the rendered height of the text object.
/// </summary>
/// <returns></returns>
protected float GetRenderedHeight()
{
return GetRenderedValues().y;
}
/// <summary>
/// Method returning the rendered height of the text object.
/// </summary>
/// <returns></returns>
protected float GetRenderedHeight(bool onlyVisibleCharacters)
{
return GetRenderedValues(onlyVisibleCharacters).y;
}
/// <summary>
/// Method to calculate the preferred width and height of the text object.
/// </summary>
/// <returns></returns>
protected virtual Vector2 CalculatePreferredValues(float defaultFontSize, Vector2 marginSize, bool ignoreTextAutoSizing)
{
//Debug.Log("*** CalculatePreferredValues() ***"); // ***** Frame: " + Time.frameCount);
////Profiler.BeginSample("TMP Generate Text - Phase I");
// Early exit if no font asset was assigned. This should not be needed since LiberationSans SDF will be assigned by default.
if (m_fontAsset == null || m_fontAsset.characterDictionary == null)
{
Debug.LogWarning("Can't Generate Mesh! No Font Asset has been assigned to Object ID: " + this.GetInstanceID());
return Vector2.zero;
}
// Early exit if we don't have any Text to generate.
if (m_char_buffer == null || m_char_buffer.Length == 0 || m_char_buffer[0] == (char)0)
{
return Vector2.zero;
}
m_currentFontAsset = m_fontAsset;
m_currentMaterial = m_sharedMaterial;
m_currentMaterialIndex = 0;
m_materialReferenceStack.SetDefault(new MaterialReference(0, m_currentFontAsset, null, m_currentMaterial, m_padding));
// Total character count is computed when the text is parsed.
int totalCharacterCount = m_totalCharacterCount; // m_VisibleCharacters.Count;
if (m_internalCharacterInfo == null || totalCharacterCount > m_internalCharacterInfo.Length)
{
m_internalCharacterInfo = new TMP_CharacterInfo[totalCharacterCount > 1024 ? totalCharacterCount + 256 : Mathf.NextPowerOfTwo(totalCharacterCount)];
}
// Calculate the scale of the font based on selected font size and sampling point size.
// baseScale is calculated using the font asset assigned to the text object.
float baseScale = m_fontScale = (defaultFontSize / m_fontAsset.fontInfo.PointSize * m_fontAsset.fontInfo.Scale * (m_isOrthographic ? 1 : 0.1f));
float currentElementScale = baseScale;
m_fontScaleMultiplier = 1;
m_currentFontSize = defaultFontSize;
m_sizeStack.SetDefault(m_currentFontSize);
float fontSizeDelta = 0;
int charCode = 0; // Holds the character code of the currently being processed character.
m_style = m_fontStyle; // Set the default style.
m_lineJustification = m_textAlignment; // Sets the line justification mode to match editor alignment.
m_lineJustificationStack.SetDefault(m_lineJustification);
float bold_xAdvance_multiplier = 1; // Used to increase spacing between character when style is bold.
m_baselineOffset = 0; // Used by subscript characters.
m_baselineOffsetStack.Clear();
m_lineOffset = 0; // Amount of space between lines (font line spacing + m_linespacing).
m_lineHeight = TMP_Math.FLOAT_UNSET;
float lineGap = m_currentFontAsset.fontInfo.LineHeight - (m_currentFontAsset.fontInfo.Ascender - m_currentFontAsset.fontInfo.Descender);
m_cSpacing = 0; // Amount of space added between characters as a result of the use of the <cspace> tag.
m_monoSpacing = 0;
float lineOffsetDelta = 0;
m_xAdvance = 0; // Used to track the position of each character.
float maxXAdvance = 0; // Used to determine Preferred Width.
tag_LineIndent = 0; // Used for indentation of text.
tag_Indent = 0;
m_indentStack.SetDefault(0);
tag_NoParsing = false;
//m_isIgnoringAlignment = false;
m_characterCount = 0; // Total characters in the char[]
// Tracking of line information
m_firstCharacterOfLine = 0;
m_maxLineAscender = k_LargeNegativeFloat;
m_maxLineDescender = k_LargePositiveFloat;
m_lineNumber = 0;
float marginWidth = marginSize.x;
//float marginHeight = marginSize.y;
m_marginLeft = 0;
m_marginRight = 0;
m_width = -1;
// Used by Unity's Auto Layout system.
float renderedWidth = 0;
float renderedHeight = 0;
float linebreakingWidth = 0;
m_isCalculatingPreferredValues = true;
// Tracking of the highest Ascender
m_maxAscender = 0;
m_maxDescender = 0;
// Initialize struct to track states of word wrapping
bool isFirstWord = true;
bool isLastBreakingChar = false;
WordWrapState savedLineState = new WordWrapState();
SaveWordWrappingState(ref savedLineState, 0, 0);
WordWrapState savedWordWrapState = new WordWrapState();
int wrappingIndex = 0;
// Counter to prevent recursive lockup when computing preferred values.
m_recursiveCount += 1;
int endTagIndex = 0;
// Parse through Character buffer to read HTML tags and begin creating mesh.
for (int i = 0; m_char_buffer[i] != 0; i++)
{
charCode = m_char_buffer[i];
// Parse Rich Text Tag
#region Parse Rich Text Tag
if (m_isRichText && charCode == 60) // '<'
{
m_isParsingText = true;
m_textElementType = TMP_TextElementType.Character;
// Check if Tag is valid. If valid, skip to the end of the validated tag.
if (ValidateHtmlTag(m_char_buffer, i + 1, out endTagIndex))
{
i = endTagIndex;
// Continue to next character or handle the sprite element
if (m_textElementType == TMP_TextElementType.Character)
continue;
}
}
else
{
m_textElementType = m_textInfo.characterInfo[m_characterCount].elementType;
m_currentMaterialIndex = m_textInfo.characterInfo[m_characterCount].materialReferenceIndex;
m_currentFontAsset = m_textInfo.characterInfo[m_characterCount].fontAsset;
}
#endregion End Parse Rich Text Tag
int prev_MaterialIndex = m_currentMaterialIndex;
bool isUsingAltTypeface = m_textInfo.characterInfo[m_characterCount].isUsingAlternateTypeface;
m_isParsingText = false;
// Handle Font Styles like LowerCase, UpperCase and SmallCaps.
#region Handling of LowerCase, UpperCase and SmallCaps Font Styles
float smallCapsMultiplier = 1.0f;
if (m_textElementType == TMP_TextElementType.Character)
{
if ((m_style & FontStyles.UpperCase) == FontStyles.UpperCase)
{
// If this character is lowercase, switch to uppercase.
if (char.IsLower((char)charCode))
charCode = char.ToUpper((char)charCode);
}
else if ((m_style & FontStyles.LowerCase) == FontStyles.LowerCase)
{
// If this character is uppercase, switch to lowercase.
if (char.IsUpper((char)charCode))
charCode = char.ToLower((char)charCode);
}
else if ((m_fontStyle & FontStyles.SmallCaps) == FontStyles.SmallCaps || (m_style & FontStyles.SmallCaps) == FontStyles.SmallCaps)
{
if (char.IsLower((char)charCode))
{
smallCapsMultiplier = 0.8f;
charCode = char.ToUpper((char)charCode);
}
}
}
#endregion
// Look up Character Data from Dictionary and cache it.
#region Look up Character Data
if (m_textElementType == TMP_TextElementType.Sprite)
{
// If a sprite is used as a fallback then get a reference to it and set the color to white.
m_currentSpriteAsset = m_textInfo.characterInfo[m_characterCount].spriteAsset;
m_spriteIndex = m_textInfo.characterInfo[m_characterCount].spriteIndex;
TMP_Sprite sprite = m_currentSpriteAsset.spriteInfoList[m_spriteIndex];
if (sprite == null) continue;
// Sprites are assigned in the E000 Private Area + sprite Index
if (charCode == 60)
charCode = 57344 + m_spriteIndex;
m_currentFontAsset = m_fontAsset;
// The sprite scale calculations are based on the font asset assigned to the text object.
// Sprite scale is used to determine line height
// Current element scale represents a modified scale to normalize the sprite based on the font baseline to ascender.
float spriteScale = (m_currentFontSize / m_fontAsset.fontInfo.PointSize * m_fontAsset.fontInfo.Scale * (m_isOrthographic ? 1 : 0.1f));
currentElementScale = m_fontAsset.fontInfo.Ascender / sprite.height * sprite.scale * spriteScale;
m_cached_TextElement = sprite;
m_internalCharacterInfo[m_characterCount].elementType = TMP_TextElementType.Sprite;
m_internalCharacterInfo[m_characterCount].scale = spriteScale;
//m_internalCharacterInfo[m_characterCount].spriteAsset = m_currentSpriteAsset;
//m_internalCharacterInfo[m_characterCount].fontAsset = m_currentFontAsset;
//m_internalCharacterInfo[m_characterCount].materialReferenceIndex = m_currentMaterialIndex;
m_currentMaterialIndex = prev_MaterialIndex;
}
else if (m_textElementType == TMP_TextElementType.Character)
{
m_cached_TextElement = m_textInfo.characterInfo[m_characterCount].textElement;
if (m_cached_TextElement == null) continue;
//m_currentFontAsset = m_textInfo.characterInfo[m_characterCount].fontAsset;
//m_currentMaterial = m_textInfo.characterInfo[m_characterCount].material;
m_currentMaterialIndex = m_textInfo.characterInfo[m_characterCount].materialReferenceIndex;
// Re-calculate font scale as the font asset may have changed.
m_fontScale = m_currentFontSize * smallCapsMultiplier / m_currentFontAsset.fontInfo.PointSize * m_currentFontAsset.fontInfo.Scale * (m_isOrthographic ? 1 : 0.1f);
currentElementScale = m_fontScale * m_fontScaleMultiplier * m_cached_TextElement.scale;
m_internalCharacterInfo[m_characterCount].elementType = TMP_TextElementType.Character;
}
#endregion
// Handle Soft Hyphen
#region Handle Soft Hyphen
float old_scale = currentElementScale;
if (charCode == 0xAD)
{
currentElementScale = 0;
}
#endregion
// Store some of the text object's information
m_internalCharacterInfo[m_characterCount].character = (char)charCode;
// Handle Kerning if Enabled.
#region Handle Kerning
GlyphValueRecord glyphAdjustments = new GlyphValueRecord();
if (m_enableKerning)
{
KerningPair adjustmentPair = null;
if (m_characterCount < totalCharacterCount - 1)
{
uint nextGlyph = m_textInfo.characterInfo[m_characterCount + 1].character;
KerningPairKey keyValue = new KerningPairKey((uint)charCode, nextGlyph);
m_currentFontAsset.kerningDictionary.TryGetValue((int)keyValue.key, out adjustmentPair);
if (adjustmentPair != null)
glyphAdjustments = adjustmentPair.firstGlyphAdjustments;
}
if (m_characterCount >= 1)
{
uint previousGlyph = m_textInfo.characterInfo[m_characterCount - 1].character;
KerningPairKey keyValue = new KerningPairKey(previousGlyph, (uint)charCode);
m_currentFontAsset.kerningDictionary.TryGetValue((int)keyValue.key, out adjustmentPair);
if (adjustmentPair != null)
glyphAdjustments += adjustmentPair.secondGlyphAdjustments;
}
}
#endregion
// Initial Implementation for RTL support.
#region Handle Right-to-Left
//if (m_isRightToLeft)
//{
// m_xAdvance -= ((m_cached_TextElement.xAdvance * bold_xAdvance_multiplier + m_characterSpacing + m_wordSpacing + m_currentFontAsset.normalSpacingOffset) * currentElementScale + m_cSpacing) * (1 - m_charWidthAdjDelta);
// if (char.IsWhiteSpace((char)charCode) || charCode == 0x200B)
// m_xAdvance -= m_wordSpacing * currentElementScale;
//}
#endregion
// Handle Mono Spacing
#region Handle Mono Spacing
float monoAdvance = 0;
if (m_monoSpacing != 0)
{
monoAdvance = (m_monoSpacing / 2 - (m_cached_TextElement.width / 2 + m_cached_TextElement.xOffset) * currentElementScale);
m_xAdvance += monoAdvance;
}
#endregion
// Set Padding based on selected font style
#region Handle Style Padding
if (m_textElementType == TMP_TextElementType.Character && !isUsingAltTypeface && ((m_style & FontStyles.Bold) == FontStyles.Bold || (m_fontStyle & FontStyles.Bold) == FontStyles.Bold)) // Checks for any combination of Bold Style.
{
//style_padding = m_currentFontAsset.boldStyle * 2;
bold_xAdvance_multiplier = 1 + m_currentFontAsset.boldSpacing * 0.01f;
}
else
{
//style_padding = m_currentFontAsset.normalStyle * 2;
bold_xAdvance_multiplier = 1.0f;
}
#endregion Handle Style Padding
m_internalCharacterInfo[m_characterCount].baseLine = 0 - m_lineOffset + m_baselineOffset;
// Compute and save text element Ascender and maximum line Ascender.
float elementAscender = m_currentFontAsset.fontInfo.Ascender * (m_textElementType == TMP_TextElementType.Character ? currentElementScale / smallCapsMultiplier : m_internalCharacterInfo[m_characterCount].scale) + m_baselineOffset;
m_internalCharacterInfo[m_characterCount].ascender = elementAscender - m_lineOffset;
m_maxLineAscender = elementAscender > m_maxLineAscender ? elementAscender : m_maxLineAscender;
// Compute and save text element Descender and maximum line Descender.
float elementDescender = m_currentFontAsset.fontInfo.Descender * (m_textElementType == TMP_TextElementType.Character ? currentElementScale / smallCapsMultiplier: m_internalCharacterInfo[m_characterCount].scale) + m_baselineOffset;
float elementDescenderII = m_internalCharacterInfo[m_characterCount].descender = elementDescender - m_lineOffset;
m_maxLineDescender = elementDescender < m_maxLineDescender ? elementDescender : m_maxLineDescender;
// Adjust maxLineAscender and maxLineDescender if style is superscript or subscript
if ((m_style & FontStyles.Subscript) == FontStyles.Subscript || (m_style & FontStyles.Superscript) == FontStyles.Superscript)
{
float baseAscender = (elementAscender - m_baselineOffset) / m_currentFontAsset.fontInfo.SubSize;
elementAscender = m_maxLineAscender;
m_maxLineAscender = baseAscender > m_maxLineAscender ? baseAscender : m_maxLineAscender;
float baseDescender = (elementDescender - m_baselineOffset) / m_currentFontAsset.fontInfo.SubSize;
elementDescender = m_maxLineDescender;
m_maxLineDescender = baseDescender < m_maxLineDescender ? baseDescender : m_maxLineDescender;
}
if (m_lineNumber == 0) m_maxAscender = m_maxAscender > elementAscender ? m_maxAscender : elementAscender;
//if (m_lineOffset == 0) pageAscender = pageAscender > elementAscender ? pageAscender : elementAscender;
// Setup Mesh for visible text elements. ie. not a SPACE / LINEFEED / CARRIAGE RETURN.
#region Handle Visible Characters
if (charCode == 9 || charCode == 0xA0 || charCode == 0x2007 || (!char.IsWhiteSpace((char)charCode) && charCode != 0x200B) || m_textElementType == TMP_TextElementType.Sprite)
{
// Check if Character exceeds the width of the Text Container
#region Handle Line Breaking, Text Auto-Sizing and Horizontal Overflow
float width = m_width != -1 ? Mathf.Min(marginWidth + 0.0001f - m_marginLeft - m_marginRight, m_width) : marginWidth + 0.0001f - m_marginLeft - m_marginRight;
bool isJustifiedOrFlush = ((_HorizontalAlignmentOptions)m_lineJustification & _HorizontalAlignmentOptions.Flush) == _HorizontalAlignmentOptions.Flush || ((_HorizontalAlignmentOptions)m_lineJustification & _HorizontalAlignmentOptions.Justified) == _HorizontalAlignmentOptions.Justified;
// Calculate the line breaking width of the text.
linebreakingWidth = m_xAdvance + m_cached_TextElement.xAdvance * (1 - m_charWidthAdjDelta) * (charCode != 0xAD ? currentElementScale : old_scale);
// Check if Character exceeds the width of the Text Container
if (linebreakingWidth > width * (isJustifiedOrFlush ? 1.05f : 1.0f))
{
// Word Wrapping
#region Handle Word Wrapping
if (enableWordWrapping && m_characterCount != m_firstCharacterOfLine)
{
// Check if word wrapping is still possible
#region Line Breaking Check
if (wrappingIndex == savedWordWrapState.previous_WordBreak || isFirstWord)
{
// Word wrapping is no longer possible. Shrink size of text if auto-sizing is enabled.
#region Text Auto-Sizing
if (ignoreTextAutoSizing == false && m_currentFontSize > m_fontSizeMin)
{
// Handle Character Width Adjustments
#region Character Width Adjustments
if (m_charWidthAdjDelta < m_charWidthMaxAdj / 100)
{
m_recursiveCount = 0;
m_charWidthAdjDelta += 0.01f;
return CalculatePreferredValues(defaultFontSize, marginSize, false);
}
#endregion
// Adjust Point Size
m_maxFontSize = defaultFontSize;
defaultFontSize -= Mathf.Max((defaultFontSize - m_minFontSize) / 2, 0.05f);
defaultFontSize = (int)(Mathf.Max(defaultFontSize, m_fontSizeMin) * 20 + 0.5f) / 20f;
if (m_recursiveCount > 20) return new Vector2(renderedWidth, renderedHeight);
return CalculatePreferredValues(defaultFontSize, marginSize, false);
}
#endregion
// Word wrapping is no longer possible, now breaking up individual words.
if (m_isCharacterWrappingEnabled == false)
{
m_isCharacterWrappingEnabled = true;
}
else
isLastBreakingChar = true;
}
#endregion
// Restore to previously stored state of last valid (space character or linefeed)
i = RestoreWordWrappingState(ref savedWordWrapState);
wrappingIndex = i; // Used to detect when line length can no longer be reduced.
// Handling for Soft Hyphen
if (m_char_buffer[i] == 0xAD) // && !m_isCharacterWrappingEnabled) // && ellipsisIndex != i && !m_isCharacterWrappingEnabled)
{
m_isTextTruncated = true;
m_char_buffer[i] = 0x2D;
return CalculatePreferredValues(defaultFontSize, marginSize, true);
}
// Check if Line Spacing of previous line needs to be adjusted.
if (m_lineNumber > 0 && !TMP_Math.Approximately(m_maxLineAscender, m_startOfLineAscender) && m_lineHeight == TMP_Math.FLOAT_UNSET)
{
//Debug.Log("(1) Adjusting Line Spacing on line #" + m_lineNumber);
float offsetDelta = m_maxLineAscender - m_startOfLineAscender;
//AdjustLineOffset(m_firstCharacterOfLine, m_characterCount, offsetDelta);
m_lineOffset += offsetDelta;
savedWordWrapState.lineOffset = m_lineOffset;
savedWordWrapState.previousLineAscender = m_maxLineAscender;
// TODO - Add check for character exceeding vertical bounds
}
//m_isNewPage = false;
// Calculate lineAscender & make sure if last character is superscript or subscript that we check that as well.
float lineAscender = m_maxLineAscender - m_lineOffset;
float lineDescender = m_maxLineDescender - m_lineOffset;
// Update maxDescender and maxVisibleDescender
m_maxDescender = m_maxDescender < lineDescender ? m_maxDescender : lineDescender;
m_firstCharacterOfLine = m_characterCount; // Store first character of the next line.
// Compute Preferred Width & Height
renderedWidth += m_xAdvance;
if (m_enableWordWrapping)
renderedHeight = m_maxAscender - m_maxDescender;
else
renderedHeight = Mathf.Max(renderedHeight, lineAscender - lineDescender);
// Store the state of the line before starting on the new line.
SaveWordWrappingState(ref savedLineState, i, m_characterCount - 1);
m_lineNumber += 1;
//isStartOfNewLine = true;
// Check to make sure Array is large enough to hold a new line.
//if (m_lineNumber >= m_internalTextInfo.lineInfo.Length)
// ResizeLineExtents(m_lineNumber);
// Apply Line Spacing based on scale of the last character of the line.
if (m_lineHeight == TMP_Math.FLOAT_UNSET)
{
float ascender = m_internalCharacterInfo[m_characterCount].ascender - m_internalCharacterInfo[m_characterCount].baseLine;
lineOffsetDelta = 0 - m_maxLineDescender + ascender + (lineGap + m_lineSpacing + m_lineSpacingDelta) * baseScale;
m_lineOffset += lineOffsetDelta;
m_startOfLineAscender = ascender;
}
else
m_lineOffset += m_lineHeight + m_lineSpacing * baseScale;
m_maxLineAscender = k_LargeNegativeFloat;
m_maxLineDescender = k_LargePositiveFloat;
m_xAdvance = 0 + tag_Indent;
continue;
}
#endregion End Word Wrapping
// Text Auto-Sizing (text exceeding Width of container.
#region Handle Text Auto-Sizing
if (ignoreTextAutoSizing == false && defaultFontSize > m_fontSizeMin)
{
// Handle Character Width Adjustments
#region Character Width Adjustments
if (m_charWidthAdjDelta < m_charWidthMaxAdj / 100)
{
m_recursiveCount = 0;
m_charWidthAdjDelta += 0.01f;
return CalculatePreferredValues(defaultFontSize, marginSize, false);
}
#endregion
// Adjust Point Size
m_maxFontSize = defaultFontSize;
defaultFontSize -= Mathf.Max((defaultFontSize - m_minFontSize) / 2, 0.05f);
defaultFontSize = (int)(Mathf.Max(defaultFontSize, m_fontSizeMin) * 20 + 0.5f) / 20f;
if (m_recursiveCount > 20) return new Vector2(renderedWidth, renderedHeight);
return CalculatePreferredValues(defaultFontSize, marginSize, false);
}
#endregion End Text Auto-Sizing
}
#endregion End Check for Characters Exceeding Width of Text Container
}
#endregion Handle Visible Characters
// Check if Line Spacing of previous line needs to be adjusted.
#region Adjust Line Spacing
if (m_lineNumber > 0 && !TMP_Math.Approximately(m_maxLineAscender, m_startOfLineAscender) && m_lineHeight == TMP_Math.FLOAT_UNSET && !m_isNewPage)
{
//Debug.Log("Inline - Adjusting Line Spacing on line #" + m_lineNumber);
//float gap = 0; // Compute gap.
float offsetDelta = m_maxLineAscender - m_startOfLineAscender;
//AdjustLineOffset(m_firstCharacterOfLine, m_characterCount, offsetDelta);
elementDescenderII -= offsetDelta;
m_lineOffset += offsetDelta;
m_startOfLineAscender += offsetDelta;
savedWordWrapState.lineOffset = m_lineOffset;
savedWordWrapState.previousLineAscender = m_startOfLineAscender;
}
#endregion
// Check if text Exceeds the vertical bounds of the margin area.
#region Check Vertical Bounds & Auto-Sizing
/*
if (m_maxAscender - elementDescenderII > marginHeight + 0.0001f)
{
// Handle Line spacing adjustments
#region Line Spacing Adjustments
if (m_enableAutoSizing && m_lineSpacingDelta > m_lineSpacingMax && m_lineNumber > 0)
{
//loopCountA = 0;
//m_lineSpacingDelta -= 1;
//GenerateTextMesh();
//return;
}
#endregion
// Handle Text Auto-sizing resulting from text exceeding vertical bounds.
#region Text Auto-Sizing (Text greater than vertical bounds)
if (m_enableAutoSizing && m_fontSize > m_fontSizeMin)
{
m_maxFontSize = m_fontSize;
m_fontSize -= Mathf.Max((m_fontSize - m_minFontSize) / 2, 0.05f);
m_fontSize = (int)(Mathf.Max(m_fontSize, m_fontSizeMin) * 20 + 0.5f) / 20f;
//m_recursiveCount = 0;
//if (loopCountA > 20) return; // Added to debug
CalculatePreferredValues(m_fontSize, marginSize, false);
return Vector2.zero;
}
#endregion Text Auto-Sizing
}
*/
#endregion Check Vertical Bounds
// Handle xAdvance & Tabulation Stops. Tab stops at every 25% of Font Size.
#region XAdvance, Tabulation & Stops
if (charCode == 9)
{
float tabSize = m_currentFontAsset.fontInfo.TabWidth * currentElementScale;
float tabs = Mathf.Ceil(m_xAdvance / tabSize) * tabSize;
m_xAdvance = tabs > m_xAdvance ? tabs : m_xAdvance + tabSize;
}
else if (m_monoSpacing != 0)
{
m_xAdvance += (m_monoSpacing - monoAdvance + ((m_characterSpacing + m_currentFontAsset.normalSpacingOffset) * currentElementScale) + m_cSpacing) * (1 - m_charWidthAdjDelta);
if (char.IsWhiteSpace((char)charCode) || charCode == 0x200B)
m_xAdvance += m_wordSpacing * currentElementScale;
}
else
{
m_xAdvance += ((m_cached_TextElement.xAdvance * bold_xAdvance_multiplier + m_characterSpacing + m_currentFontAsset.normalSpacingOffset + glyphAdjustments.xAdvance) * currentElementScale + m_cSpacing) * (1 - m_charWidthAdjDelta);
if (char.IsWhiteSpace((char)charCode) || charCode == 0x200B)
m_xAdvance += m_wordSpacing * currentElementScale;
}
#endregion Tabulation & Stops
// Handle Carriage Return
#region Carriage Return
if (charCode == 13)
{
maxXAdvance = Mathf.Max(maxXAdvance, renderedWidth + m_xAdvance);
renderedWidth = 0;
m_xAdvance = 0 + tag_Indent;
}
#endregion Carriage Return
// Handle Line Spacing Adjustments + Word Wrapping & special case for last line.
#region Check for Line Feed and Last Character
if (charCode == 10 || m_characterCount == totalCharacterCount - 1)
{
// Check if Line Spacing of previous line needs to be adjusted.
if (m_lineNumber > 0 && !TMP_Math.Approximately(m_maxLineAscender, m_startOfLineAscender) && m_lineHeight == TMP_Math.FLOAT_UNSET)
{
//Debug.Log("(2) Adjusting Line Spacing on line #" + m_lineNumber);
float offsetDelta = m_maxLineAscender - m_startOfLineAscender;
//AdjustLineOffset(m_firstCharacterOfLine, m_characterCount, offsetDelta);
elementDescenderII -= offsetDelta;
m_lineOffset += offsetDelta;
}
// Calculate lineAscender & make sure if last character is superscript or subscript that we check that as well.
//float lineAscender = m_maxLineAscender - m_lineOffset;
float lineDescender = m_maxLineDescender - m_lineOffset;
// Update maxDescender and maxVisibleDescender
m_maxDescender = m_maxDescender < lineDescender ? m_maxDescender : lineDescender;
m_firstCharacterOfLine = m_characterCount + 1;
// Store PreferredWidth paying attention to linefeed and last character of text.
if (charCode == 10 && m_characterCount != totalCharacterCount - 1)
{
maxXAdvance = Mathf.Max(maxXAdvance, renderedWidth + linebreakingWidth);
renderedWidth = 0;
}
else
renderedWidth = Mathf.Max(maxXAdvance, renderedWidth + linebreakingWidth);
renderedHeight = m_maxAscender - m_maxDescender;
// Add new line if not last lines or character.
if (charCode == 10)
{
// Store the state of the line before starting on the new line.
SaveWordWrappingState(ref savedLineState, i, m_characterCount);
// Store the state of the last Character before the new line.
SaveWordWrappingState(ref savedWordWrapState, i, m_characterCount);
m_lineNumber += 1;
// Apply Line Spacing
if (m_lineHeight == TMP_Math.FLOAT_UNSET)
{
lineOffsetDelta = 0 - m_maxLineDescender + elementAscender + (lineGap + m_lineSpacing + m_paragraphSpacing + m_lineSpacingDelta) * baseScale;
m_lineOffset += lineOffsetDelta;
}
else
m_lineOffset += m_lineHeight + (m_lineSpacing + m_paragraphSpacing) * baseScale;
m_maxLineAscender = k_LargeNegativeFloat;
m_maxLineDescender = k_LargePositiveFloat;
m_startOfLineAscender = elementAscender;
m_xAdvance = 0 + tag_LineIndent + tag_Indent;
m_characterCount += 1;
continue;
}
}
#endregion Check for Linefeed or Last Character
// Save State of Mesh Creation for handling of Word Wrapping
#region Save Word Wrapping State
if (m_enableWordWrapping || m_overflowMode == TextOverflowModes.Truncate || m_overflowMode == TextOverflowModes.Ellipsis)
{
if ((char.IsWhiteSpace((char)charCode) || charCode == 0x200B || charCode == 0x2D || charCode == 0xAD) && !m_isNonBreakingSpace && charCode != 0xA0 && charCode != 0x2011 && charCode != 0x202F && charCode != 0x2060)
{
// We store the state of numerous variables for the most recent Space, LineFeed or Carriage Return to enable them to be restored
// for Word Wrapping.
SaveWordWrappingState(ref savedWordWrapState, i, m_characterCount);
m_isCharacterWrappingEnabled = false;
isFirstWord = false;
}
// Handling for East Asian languages
else if ((charCode > 0x1100 && charCode < 0x11ff || /* Hangul Jamo */
charCode > 0x2E80 && charCode < 0x9FFF || /* CJK */
charCode > 0xA960 && charCode < 0xA97F || /* Hangul Jame Extended-A */
charCode > 0xAC00 && charCode < 0xD7FF || /* Hangul Syllables */
charCode > 0xF900 && charCode < 0xFAFF || /* CJK Compatibility Ideographs */
charCode > 0xFE30 && charCode < 0xFE4F || /* CJK Compatibility Forms */
charCode > 0xFF00 && charCode < 0xFFEF) /* CJK Halfwidth */
&& !m_isNonBreakingSpace)
{
if (isFirstWord || isLastBreakingChar || TMP_Settings.linebreakingRules.leadingCharacters.ContainsKey(charCode) == false &&
(m_characterCount < totalCharacterCount - 1 &&
TMP_Settings.linebreakingRules.followingCharacters.ContainsKey(m_internalCharacterInfo[m_characterCount + 1].character) == false))
{
SaveWordWrappingState(ref savedWordWrapState, i, m_characterCount);
m_isCharacterWrappingEnabled = false;
isFirstWord = false;
}
}
else if ((isFirstWord || m_isCharacterWrappingEnabled == true || isLastBreakingChar))
SaveWordWrappingState(ref savedWordWrapState, i, m_characterCount);
}
#endregion Save Word Wrapping State
m_characterCount += 1;
}
// Check Auto Sizing and increase font size to fill text container.
#region Check Auto-Sizing (Upper Font Size Bounds)
fontSizeDelta = m_maxFontSize - m_minFontSize;
if (!m_isCharacterWrappingEnabled && ignoreTextAutoSizing == false && fontSizeDelta > 0.051f && defaultFontSize < m_fontSizeMax)
{
m_minFontSize = defaultFontSize;
defaultFontSize += Mathf.Max((m_maxFontSize - defaultFontSize) / 2, 0.05f);
defaultFontSize = (int)(Mathf.Min(defaultFontSize, m_fontSizeMax) * 20 + 0.5f) / 20f;
if (m_recursiveCount > 20) return new Vector2(renderedWidth, renderedHeight);
return CalculatePreferredValues(defaultFontSize, marginSize, false);
}
#endregion End Auto-sizing Check
m_isCharacterWrappingEnabled = false;
m_isCalculatingPreferredValues = false;
// Adjust Preferred Width and Height to account for Margins.
renderedWidth += m_margin.x > 0 ? m_margin.x : 0;
renderedWidth += m_margin.z > 0 ? m_margin.z : 0;
renderedHeight += m_margin.y > 0 ? m_margin.y : 0;
renderedHeight += m_margin.w > 0 ? m_margin.w : 0;
// Round Preferred Values to nearest 5/100.
renderedWidth = (int)(renderedWidth * 100 + 1f) / 100f;
renderedHeight = (int)(renderedHeight * 100 + 1f) / 100f;
//Debug.Log("Preferred Values: (" + renderedWidth + ", " + renderedHeight + ") with Recursive count of " + m_recursiveCount);
////Profiler.EndSample();
return new Vector2(renderedWidth, renderedHeight);
}
/// <summary>
/// Method returning the compound bounds of the text object and child sub objects.
/// </summary>
/// <returns></returns>
protected virtual Bounds GetCompoundBounds() { return new Bounds(); }
/// <summary>
/// Method which returns the bounds of the text object;
/// </summary>
/// <returns></returns>
protected Bounds GetTextBounds()
{
if (m_textInfo == null || m_textInfo.characterCount > m_textInfo.characterInfo.Length) return new Bounds();
Extents extent = new Extents(k_LargePositiveVector2, k_LargeNegativeVector2);
for (int i = 0; i < m_textInfo.characterCount && i < m_textInfo.characterInfo.Length; i++)
{
if (!m_textInfo.characterInfo[i].isVisible) continue;
extent.min.x = Mathf.Min(extent.min.x, m_textInfo.characterInfo[i].bottomLeft.x);
extent.min.y = Mathf.Min(extent.min.y, m_textInfo.characterInfo[i].descender);
extent.max.x = Mathf.Max(extent.max.x, m_textInfo.characterInfo[i].xAdvance);
extent.max.y = Mathf.Max(extent.max.y, m_textInfo.characterInfo[i].ascender);
}
Vector2 size;
size.x = extent.max.x - extent.min.x;
size.y = extent.max.y - extent.min.y;
Vector3 center = (extent.min + extent.max) / 2;
return new Bounds(center, size);
}
/// <summary>
/// Method which returns the bounds of the text object;
/// </summary>
/// <param name="onlyVisibleCharacters"></param>
/// <returns></returns>
protected Bounds GetTextBounds(bool onlyVisibleCharacters)
{
if (m_textInfo == null) return new Bounds();
Extents extent = new Extents(k_LargePositiveVector2, k_LargeNegativeVector2);
for (int i = 0; i < m_textInfo.characterCount; i++)
{
if ((i > maxVisibleCharacters || m_textInfo.characterInfo[i].lineNumber > m_maxVisibleLines) && onlyVisibleCharacters) break;
if (onlyVisibleCharacters && !m_textInfo.characterInfo[i].isVisible) continue;
extent.min.x = Mathf.Min(extent.min.x, m_textInfo.characterInfo[i].origin);
extent.min.y = Mathf.Min(extent.min.y, m_textInfo.characterInfo[i].descender);
extent.max.x = Mathf.Max(extent.max.x, m_textInfo.characterInfo[i].xAdvance);
extent.max.y = Mathf.Max(extent.max.y, m_textInfo.characterInfo[i].ascender);
}
Vector2 size;
size.x = extent.max.x - extent.min.x;
size.y = extent.max.y - extent.min.y;
Vector2 center = (extent.min + extent.max) / 2;
return new Bounds(center, size);
}
/// <summary>
/// Method to adjust line spacing as a result of using different fonts or font point size.
/// </summary>
/// <param name="startIndex"></param>
/// <param name="endIndex"></param>
/// <param name="offset"></param>
protected virtual void AdjustLineOffset(int startIndex, int endIndex, float offset) { }
/// <summary>
/// Function to increase the size of the Line Extents Array.
/// </summary>
/// <param name="size"></param>
protected void ResizeLineExtents(int size)
{
size = size > 1024 ? size + 256 : Mathf.NextPowerOfTwo(size + 1);
TMP_LineInfo[] temp_lineInfo = new TMP_LineInfo[size];
for (int i = 0; i < size; i++)
{
if (i < m_textInfo.lineInfo.Length)
temp_lineInfo[i] = m_textInfo.lineInfo[i];
else
{
temp_lineInfo[i].lineExtents.min = k_LargePositiveVector2;
temp_lineInfo[i].lineExtents.max = k_LargeNegativeVector2;
temp_lineInfo[i].ascender = k_LargeNegativeFloat;
temp_lineInfo[i].descender = k_LargePositiveFloat;
}
}
m_textInfo.lineInfo = temp_lineInfo;
}
protected static Vector2 k_LargePositiveVector2 = new Vector2(TMP_Math.INT_MAX, TMP_Math.INT_MAX);
protected static Vector2 k_LargeNegativeVector2 = new Vector2(TMP_Math.INT_MIN, TMP_Math.INT_MIN);
protected static float k_LargePositiveFloat = TMP_Math.FLOAT_MAX;
protected static float k_LargeNegativeFloat = TMP_Math.FLOAT_MIN;
protected static int k_LargePositiveInt = TMP_Math.INT_MAX;
protected static int k_LargeNegativeInt = TMP_Math.INT_MIN;
/// <summary>
/// Function used to evaluate the length of a text string.
/// </summary>
/// <param name="text"></param>
/// <returns></returns>
public virtual TMP_TextInfo GetTextInfo(string text) { return null; }
/// <summary>
/// Function to force an update of the margin size.
/// </summary>
public virtual void ComputeMarginSize() { }
/// <summary>
/// Function used in conjunction with GetTextInfo to figure out Array allocations.
/// </summary>
/// <param name="chars"></param>
/// <returns></returns>
//protected int GetArraySizes(int[] chars)
//{
// //Debug.Log("Set Array Size called.");
// //int visibleCount = 0;
// //int totalCount = 0;
// int tagEnd = 0;
// m_totalCharacterCount = 0;
// m_isUsingBold = false;
// m_isParsingText = false;
// //m_VisibleCharacters.Clear();
// for (int i = 0; chars[i] != 0; i++)
// {
// int c = chars[i];
// if (m_isRichText && c == 60) // if Char '<'
// {
// // Check if Tag is Valid
// if (ValidateHtmlTag(chars, i + 1, out tagEnd))
// {
// i = tagEnd;
// //if ((m_style & FontStyles.Underline) == FontStyles.Underline) visibleCount += 3;
// if ((m_style & FontStyles.Bold) == FontStyles.Bold) m_isUsingBold = true;
// continue;
// }
// }
// //if (!char.IsWhiteSpace((char)c) && c != 0x200B)
// //{
// //visibleCount += 1;
// //}
// //m_VisibleCharacters.Add((char)c);
// m_totalCharacterCount += 1;
// }
// return m_totalCharacterCount;
//}
/// <summary>
/// Save the State of various variables used in the mesh creation loop in conjunction with Word Wrapping
/// </summary>
/// <param name="state"></param>
/// <param name="index"></param>
/// <param name="count"></param>
protected void SaveWordWrappingState(ref WordWrapState state, int index, int count)
{
// Multi Font & Material support related
state.currentFontAsset = m_currentFontAsset;
state.currentSpriteAsset = m_currentSpriteAsset;
state.currentMaterial = m_currentMaterial;
state.currentMaterialIndex = m_currentMaterialIndex;
state.previous_WordBreak = index;
state.total_CharacterCount = count;
state.visible_CharacterCount = m_lineVisibleCharacterCount;
//state.visible_CharacterCount = m_visibleCharacterCount;
//state.visible_SpriteCount = m_visibleSpriteCount;
state.visible_LinkCount = m_textInfo.linkCount;
state.firstCharacterIndex = m_firstCharacterOfLine;
state.firstVisibleCharacterIndex = m_firstVisibleCharacterOfLine;
state.lastVisibleCharIndex = m_lastVisibleCharacterOfLine;
state.fontStyle = m_style;
state.fontScale = m_fontScale;
//state.maxFontScale = m_maxFontScale;
state.fontScaleMultiplier = m_fontScaleMultiplier;
state.currentFontSize = m_currentFontSize;
state.xAdvance = m_xAdvance;
state.maxCapHeight = m_maxCapHeight;
state.maxAscender = m_maxAscender;
state.maxDescender = m_maxDescender;
state.maxLineAscender = m_maxLineAscender;
state.maxLineDescender = m_maxLineDescender;
state.previousLineAscender = m_startOfLineAscender;
state.preferredWidth = m_preferredWidth;
state.preferredHeight = m_preferredHeight;
state.meshExtents = m_meshExtents;
state.lineNumber = m_lineNumber;
state.lineOffset = m_lineOffset;
state.baselineOffset = m_baselineOffset;
//state.alignment = m_lineJustification;
state.vertexColor = m_htmlColor;
state.underlineColor = m_underlineColor;
state.strikethroughColor = m_strikethroughColor;
state.highlightColor = m_highlightColor;
state.isNonBreakingSpace = m_isNonBreakingSpace;
state.tagNoParsing = tag_NoParsing;
// XML Tag Stack
state.basicStyleStack = m_fontStyleStack;
state.colorStack = m_colorStack;
state.underlineColorStack = m_underlineColorStack;
state.strikethroughColorStack = m_strikethroughColorStack;
state.highlightColorStack = m_highlightColorStack;
state.colorGradientStack = m_colorGradientStack;
state.sizeStack = m_sizeStack;
state.indentStack = m_indentStack;
state.fontWeightStack = m_fontWeightStack;
state.styleStack = m_styleStack;
state.baselineStack = m_baselineOffsetStack;
state.actionStack = m_actionStack;
state.materialReferenceStack = m_materialReferenceStack;
state.lineJustificationStack = m_lineJustificationStack;
//state.spriteAnimationStack = m_spriteAnimationStack;
state.spriteAnimationID = m_spriteAnimationID;
if (m_lineNumber < m_textInfo.lineInfo.Length)
state.lineInfo = m_textInfo.lineInfo[m_lineNumber];
}
/// <summary>
/// Restore the State of various variables used in the mesh creation loop.
/// </summary>
/// <param name="state"></param>
/// <returns></returns>
protected int RestoreWordWrappingState(ref WordWrapState state)
{
int index = state.previous_WordBreak;
// Multi Font & Material support related
m_currentFontAsset = state.currentFontAsset;
m_currentSpriteAsset = state.currentSpriteAsset;
m_currentMaterial = state.currentMaterial;
m_currentMaterialIndex = state.currentMaterialIndex;
m_characterCount = state.total_CharacterCount + 1;
m_lineVisibleCharacterCount = state.visible_CharacterCount;
//m_visibleCharacterCount = state.visible_CharacterCount;
//m_visibleSpriteCount = state.visible_SpriteCount;
m_textInfo.linkCount = state.visible_LinkCount;
m_firstCharacterOfLine = state.firstCharacterIndex;
m_firstVisibleCharacterOfLine = state.firstVisibleCharacterIndex;
m_lastVisibleCharacterOfLine = state.lastVisibleCharIndex;
m_style = state.fontStyle;
m_fontScale = state.fontScale;
m_fontScaleMultiplier = state.fontScaleMultiplier;
//m_maxFontScale = state.maxFontScale;
m_currentFontSize = state.currentFontSize;
m_xAdvance = state.xAdvance;
m_maxCapHeight = state.maxCapHeight;
m_maxAscender = state.maxAscender;
m_maxDescender = state.maxDescender;
m_maxLineAscender = state.maxLineAscender;
m_maxLineDescender = state.maxLineDescender;
m_startOfLineAscender = state.previousLineAscender;
m_preferredWidth = state.preferredWidth;
m_preferredHeight = state.preferredHeight;
m_meshExtents = state.meshExtents;
m_lineNumber = state.lineNumber;
m_lineOffset = state.lineOffset;
m_baselineOffset = state.baselineOffset;
//m_lineJustification = state.alignment;
m_htmlColor = state.vertexColor;
m_underlineColor = state.underlineColor;
m_strikethroughColor = state.strikethroughColor;
m_highlightColor = state.highlightColor;
m_isNonBreakingSpace = state.isNonBreakingSpace;
tag_NoParsing = state.tagNoParsing;
// XML Tag Stack
m_fontStyleStack = state.basicStyleStack;
m_colorStack = state.colorStack;
m_underlineColorStack = state.underlineColorStack;
m_strikethroughColorStack = state.strikethroughColorStack;
m_highlightColorStack = state.highlightColorStack;
m_colorGradientStack = state.colorGradientStack;
m_sizeStack = state.sizeStack;
m_indentStack = state.indentStack;
m_fontWeightStack = state.fontWeightStack;
m_styleStack = state.styleStack;
m_baselineOffsetStack = state.baselineStack;
m_actionStack = state.actionStack;
m_materialReferenceStack = state.materialReferenceStack;
m_lineJustificationStack = state.lineJustificationStack;
//m_spriteAnimationStack = state.spriteAnimationStack;
m_spriteAnimationID = state.spriteAnimationID;
if (m_lineNumber < m_textInfo.lineInfo.Length)
m_textInfo.lineInfo[m_lineNumber] = state.lineInfo;
return index;
}
/// <summary>
/// Store vertex information for each character.
/// </summary>
/// <param name="style_padding">Style_padding.</param>
/// <param name="vertexColor">Vertex color.</param>
protected virtual void SaveGlyphVertexInfo(float padding, float style_padding, Color32 vertexColor)
{
// Save the Vertex Position for the Character
#region Setup Mesh Vertices
m_textInfo.characterInfo[m_characterCount].vertex_BL.position = m_textInfo.characterInfo[m_characterCount].bottomLeft;
m_textInfo.characterInfo[m_characterCount].vertex_TL.position = m_textInfo.characterInfo[m_characterCount].topLeft;
m_textInfo.characterInfo[m_characterCount].vertex_TR.position = m_textInfo.characterInfo[m_characterCount].topRight;
m_textInfo.characterInfo[m_characterCount].vertex_BR.position = m_textInfo.characterInfo[m_characterCount].bottomRight;
#endregion
#region Setup Vertex Colors
// Alpha is the lower of the vertex color or tag color alpha used.
vertexColor.a = m_fontColor32.a < vertexColor.a ? (byte)(m_fontColor32.a) : (byte)(vertexColor.a);
// Handle Vertex Colors & Vertex Color Gradient
if (!m_enableVertexGradient)
{
m_textInfo.characterInfo[m_characterCount].vertex_BL.color = vertexColor;
m_textInfo.characterInfo[m_characterCount].vertex_TL.color = vertexColor;
m_textInfo.characterInfo[m_characterCount].vertex_TR.color = vertexColor;
m_textInfo.characterInfo[m_characterCount].vertex_BR.color = vertexColor;
}
else
{
if (!m_overrideHtmlColors && m_colorStack.index > 1)
{
m_textInfo.characterInfo[m_characterCount].vertex_BL.color = vertexColor;
m_textInfo.characterInfo[m_characterCount].vertex_TL.color = vertexColor;
m_textInfo.characterInfo[m_characterCount].vertex_TR.color = vertexColor;
m_textInfo.characterInfo[m_characterCount].vertex_BR.color = vertexColor;
}
else // Handle Vertex Color Gradient
{
// Use Vertex Color Gradient Preset (if one is assigned)
if (m_fontColorGradientPreset != null)
{
m_textInfo.characterInfo[m_characterCount].vertex_BL.color = m_fontColorGradientPreset.bottomLeft * vertexColor;
m_textInfo.characterInfo[m_characterCount].vertex_TL.color = m_fontColorGradientPreset.topLeft * vertexColor;
m_textInfo.characterInfo[m_characterCount].vertex_TR.color = m_fontColorGradientPreset.topRight * vertexColor;
m_textInfo.characterInfo[m_characterCount].vertex_BR.color = m_fontColorGradientPreset.bottomRight * vertexColor;
}
else
{
m_textInfo.characterInfo[m_characterCount].vertex_BL.color = m_fontColorGradient.bottomLeft * vertexColor;
m_textInfo.characterInfo[m_characterCount].vertex_TL.color = m_fontColorGradient.topLeft * vertexColor;
m_textInfo.characterInfo[m_characterCount].vertex_TR.color = m_fontColorGradient.topRight * vertexColor;
m_textInfo.characterInfo[m_characterCount].vertex_BR.color = m_fontColorGradient.bottomRight * vertexColor;
}
}
}
if (m_colorGradientPreset != null)
{
m_textInfo.characterInfo[m_characterCount].vertex_BL.color *= m_colorGradientPreset.bottomLeft;
m_textInfo.characterInfo[m_characterCount].vertex_TL.color *= m_colorGradientPreset.topLeft;
m_textInfo.characterInfo[m_characterCount].vertex_TR.color *= m_colorGradientPreset.topRight;
m_textInfo.characterInfo[m_characterCount].vertex_BR.color *= m_colorGradientPreset.bottomRight;
}
#endregion
// Apply style_padding only if this is a SDF Shader.
if (!m_isSDFShader)
style_padding = 0;
// Setup UVs for the Character
#region Setup UVs
FaceInfo faceInfo = m_currentFontAsset.fontInfo;
Vector2 uv0;
uv0.x = (m_cached_TextElement.x - padding - style_padding) / faceInfo.AtlasWidth;
uv0.y = 1 - (m_cached_TextElement.y + padding + style_padding + m_cached_TextElement.height) / faceInfo.AtlasHeight;
Vector2 uv1;
uv1.x = uv0.x;
uv1.y = 1 - (m_cached_TextElement.y - padding - style_padding) / faceInfo.AtlasHeight;
Vector2 uv2;
uv2.x = (m_cached_TextElement.x + padding + style_padding + m_cached_TextElement.width) / faceInfo.AtlasWidth;
uv2.y = uv1.y;
Vector2 uv3;
uv3.x = uv2.x;
uv3.y = uv0.y;
//Vector2 uv0 = new Vector2((m_cached_TextElement.x - padding - style_padding) / faceInfo.AtlasWidth, 1 - (m_cached_TextElement.y + padding + style_padding + m_cached_TextElement.height) / faceInfo.AtlasHeight); // bottom left
//Vector2 uv1 = new Vector2(uv0.x, 1 - (m_cached_TextElement.y - padding - style_padding) / faceInfo.AtlasHeight); // top left
//Vector2 uv2 = new Vector2((m_cached_TextElement.x + padding + style_padding + m_cached_TextElement.width) / faceInfo.AtlasWidth, uv1.y); // top right
//Vector2 uv3 = new Vector2(uv2.x, uv0.y); // bottom right
// Store UV Information
m_textInfo.characterInfo[m_characterCount].vertex_BL.uv = uv0;
m_textInfo.characterInfo[m_characterCount].vertex_TL.uv = uv1;
m_textInfo.characterInfo[m_characterCount].vertex_TR.uv = uv2;
m_textInfo.characterInfo[m_characterCount].vertex_BR.uv = uv3;
#endregion Setup UVs
// Normal
#region Setup Normals & Tangents
//Vector3 normal = new Vector3(0, 0, -1);
//m_textInfo.characterInfo[m_characterCount].vertex_BL.normal = normal;
//m_textInfo.characterInfo[m_characterCount].vertex_TL.normal = normal;
//m_textInfo.characterInfo[m_characterCount].vertex_TR.normal = normal;
//m_textInfo.characterInfo[m_characterCount].vertex_BR.normal = normal;
// Tangents
//Vector4 tangent = new Vector4(-1, 0, 0, 1);
//m_textInfo.characterInfo[m_characterCount].vertex_BL.tangent = tangent;
//m_textInfo.characterInfo[m_characterCount].vertex_TL.tangent = tangent;
//m_textInfo.characterInfo[m_characterCount].vertex_TR.tangent = tangent;
//m_textInfo.characterInfo[m_characterCount].vertex_BR.tangent = tangent;
#endregion end Normals & Tangents
}
/// <summary>
/// Store vertex information for each sprite.
/// </summary>
/// <param name="padding"></param>
/// <param name="style_padding"></param>
/// <param name="vertexColor"></param>
protected virtual void SaveSpriteVertexInfo(Color32 vertexColor)
{
// Save the Vertex Position for the Character
#region Setup Mesh Vertices
m_textInfo.characterInfo[m_characterCount].vertex_BL.position = m_textInfo.characterInfo[m_characterCount].bottomLeft;
m_textInfo.characterInfo[m_characterCount].vertex_TL.position = m_textInfo.characterInfo[m_characterCount].topLeft;
m_textInfo.characterInfo[m_characterCount].vertex_TR.position = m_textInfo.characterInfo[m_characterCount].topRight;
m_textInfo.characterInfo[m_characterCount].vertex_BR.position = m_textInfo.characterInfo[m_characterCount].bottomRight;
#endregion
// Vertex Color Alpha
if (m_tintAllSprites) m_tintSprite = true;
Color32 spriteColor = m_tintSprite ? m_spriteColor.Multiply(vertexColor) : m_spriteColor;
spriteColor.a = spriteColor.a < m_fontColor32.a ? spriteColor.a = spriteColor.a < vertexColor.a ? spriteColor.a : vertexColor.a : m_fontColor32.a;
Color32 c0 = spriteColor;
Color32 c1 = spriteColor;
Color32 c2 = spriteColor;
Color32 c3 = spriteColor;
if (m_enableVertexGradient)
{
if (m_fontColorGradientPreset != null)
{
c0 = m_tintSprite ? c0.Multiply(m_fontColorGradientPreset.bottomLeft) : c0;
c1 = m_tintSprite ? c1.Multiply(m_fontColorGradientPreset.topLeft) : c1;
c2 = m_tintSprite ? c2.Multiply(m_fontColorGradientPreset.topRight) : c2;
c3 = m_tintSprite ? c3.Multiply(m_fontColorGradientPreset.bottomRight) : c3;
}
else
{
c0 = m_tintSprite ? c0.Multiply(m_fontColorGradient.bottomLeft) : c0;
c1 = m_tintSprite ? c1.Multiply(m_fontColorGradient.topLeft) : c1;
c2 = m_tintSprite ? c2.Multiply(m_fontColorGradient.topRight) : c2;
c3 = m_tintSprite ? c3.Multiply(m_fontColorGradient.bottomRight) : c3;
}
}
if (m_colorGradientPreset != null)
{
c0 = m_tintSprite ? c0.Multiply(m_colorGradientPreset.bottomLeft) : c0;
c1 = m_tintSprite ? c1.Multiply(m_colorGradientPreset.topLeft) : c1;
c2 = m_tintSprite ? c2.Multiply(m_colorGradientPreset.topRight) : c2;
c3 = m_tintSprite ? c3.Multiply(m_colorGradientPreset.bottomRight) : c3;
}
m_textInfo.characterInfo[m_characterCount].vertex_BL.color = c0;
m_textInfo.characterInfo[m_characterCount].vertex_TL.color = c1;
m_textInfo.characterInfo[m_characterCount].vertex_TR.color = c2;
m_textInfo.characterInfo[m_characterCount].vertex_BR.color = c3;
// Setup UVs for the Character
#region Setup UVs
Vector2 uv0 = new Vector2(m_cached_TextElement.x / m_currentSpriteAsset.spriteSheet.width, m_cached_TextElement.y / m_currentSpriteAsset.spriteSheet.height); // bottom left
Vector2 uv1 = new Vector2(uv0.x, (m_cached_TextElement.y + m_cached_TextElement.height) / m_currentSpriteAsset.spriteSheet.height); // top left
Vector2 uv2 = new Vector2((m_cached_TextElement.x + m_cached_TextElement.width) / m_currentSpriteAsset.spriteSheet.width, uv1.y); // top right
Vector2 uv3 = new Vector2(uv2.x, uv0.y); // bottom right
// Store UV Information
m_textInfo.characterInfo[m_characterCount].vertex_BL.uv = uv0;
m_textInfo.characterInfo[m_characterCount].vertex_TL.uv = uv1;
m_textInfo.characterInfo[m_characterCount].vertex_TR.uv = uv2;
m_textInfo.characterInfo[m_characterCount].vertex_BR.uv = uv3;
#endregion Setup UVs
// Normal
#region Setup Normals & Tangents
//Vector3 normal = new Vector3(0, 0, -1);
//m_textInfo.characterInfo[m_characterCount].vertex_BL.normal = normal;
//m_textInfo.characterInfo[m_characterCount].vertex_TL.normal = normal;
//m_textInfo.characterInfo[m_characterCount].vertex_TR.normal = normal;
//m_textInfo.characterInfo[m_characterCount].vertex_BR.normal = normal;
// Tangents
//Vector4 tangent = new Vector4(-1, 0, 0, 1);
//m_textInfo.characterInfo[m_characterCount].vertex_BL.tangent = tangent;
//m_textInfo.characterInfo[m_characterCount].vertex_TL.tangent = tangent;
//m_textInfo.characterInfo[m_characterCount].vertex_TR.tangent = tangent;
//m_textInfo.characterInfo[m_characterCount].vertex_BR.tangent = tangent;
#endregion end Normals & Tangents
}
/// <summary>
/// Store vertex attributes into the appropriate TMP_MeshInfo.
/// </summary>
/// <param name="i"></param>
/// <param name="index_X4"></param>
protected virtual void FillCharacterVertexBuffers(int i, int index_X4)
{
int materialIndex = m_textInfo.characterInfo[i].materialReferenceIndex;
index_X4 = m_textInfo.meshInfo[materialIndex].vertexCount;
// Make sure buffers allocation are sufficient to hold the vertex data
//if (m_textInfo.meshInfo[materialIndex].vertices.Length < index_X4 + 4)
// m_textInfo.meshInfo[materialIndex].ResizeMeshInfo(Mathf.NextPowerOfTwo(index_X4 + 4));
TMP_CharacterInfo[] characterInfoArray = m_textInfo.characterInfo;
m_textInfo.characterInfo[i].vertexIndex = index_X4;
// Setup Vertices for Characters
m_textInfo.meshInfo[materialIndex].vertices[0 + index_X4] = characterInfoArray[i].vertex_BL.position;
m_textInfo.meshInfo[materialIndex].vertices[1 + index_X4] = characterInfoArray[i].vertex_TL.position;
m_textInfo.meshInfo[materialIndex].vertices[2 + index_X4] = characterInfoArray[i].vertex_TR.position;
m_textInfo.meshInfo[materialIndex].vertices[3 + index_X4] = characterInfoArray[i].vertex_BR.position;
// Setup UVS0
m_textInfo.meshInfo[materialIndex].uvs0[0 + index_X4] = characterInfoArray[i].vertex_BL.uv;
m_textInfo.meshInfo[materialIndex].uvs0[1 + index_X4] = characterInfoArray[i].vertex_TL.uv;
m_textInfo.meshInfo[materialIndex].uvs0[2 + index_X4] = characterInfoArray[i].vertex_TR.uv;
m_textInfo.meshInfo[materialIndex].uvs0[3 + index_X4] = characterInfoArray[i].vertex_BR.uv;
// Setup UVS2
m_textInfo.meshInfo[materialIndex].uvs2[0 + index_X4] = characterInfoArray[i].vertex_BL.uv2;
m_textInfo.meshInfo[materialIndex].uvs2[1 + index_X4] = characterInfoArray[i].vertex_TL.uv2;
m_textInfo.meshInfo[materialIndex].uvs2[2 + index_X4] = characterInfoArray[i].vertex_TR.uv2;
m_textInfo.meshInfo[materialIndex].uvs2[3 + index_X4] = characterInfoArray[i].vertex_BR.uv2;
// Setup UVS4
//m_textInfo.meshInfo[0].uvs4[0 + index_X4] = characterInfoArray[i].vertex_BL.uv4;
//m_textInfo.meshInfo[0].uvs4[1 + index_X4] = characterInfoArray[i].vertex_TL.uv4;
//m_textInfo.meshInfo[0].uvs4[2 + index_X4] = characterInfoArray[i].vertex_TR.uv4;
//m_textInfo.meshInfo[0].uvs4[3 + index_X4] = characterInfoArray[i].vertex_BR.uv4;
// setup Vertex Colors
m_textInfo.meshInfo[materialIndex].colors32[0 + index_X4] = characterInfoArray[i].vertex_BL.color;
m_textInfo.meshInfo[materialIndex].colors32[1 + index_X4] = characterInfoArray[i].vertex_TL.color;
m_textInfo.meshInfo[materialIndex].colors32[2 + index_X4] = characterInfoArray[i].vertex_TR.color;
m_textInfo.meshInfo[materialIndex].colors32[3 + index_X4] = characterInfoArray[i].vertex_BR.color;
m_textInfo.meshInfo[materialIndex].vertexCount = index_X4 + 4;
}
protected virtual void FillCharacterVertexBuffers(int i, int index_X4, bool isVolumetric)
{
int materialIndex = m_textInfo.characterInfo[i].materialReferenceIndex;
index_X4 = m_textInfo.meshInfo[materialIndex].vertexCount;
TMP_CharacterInfo[] characterInfoArray = m_textInfo.characterInfo;
m_textInfo.characterInfo[i].vertexIndex = index_X4;
// Setup Vertices for Characters
m_textInfo.meshInfo[materialIndex].vertices[0 + index_X4] = characterInfoArray[i].vertex_BL.position;
m_textInfo.meshInfo[materialIndex].vertices[1 + index_X4] = characterInfoArray[i].vertex_TL.position;
m_textInfo.meshInfo[materialIndex].vertices[2 + index_X4] = characterInfoArray[i].vertex_TR.position;
m_textInfo.meshInfo[materialIndex].vertices[3 + index_X4] = characterInfoArray[i].vertex_BR.position;
if (isVolumetric)
{
Vector3 depth = new Vector3(0, 0, m_fontSize * m_fontScale);
m_textInfo.meshInfo[materialIndex].vertices[4 + index_X4] = characterInfoArray[i].vertex_BL.position + depth;
m_textInfo.meshInfo[materialIndex].vertices[5 + index_X4] = characterInfoArray[i].vertex_TL.position + depth;
m_textInfo.meshInfo[materialIndex].vertices[6 + index_X4] = characterInfoArray[i].vertex_TR.position + depth;
m_textInfo.meshInfo[materialIndex].vertices[7 + index_X4] = characterInfoArray[i].vertex_BR.position + depth;
}
// Setup UVS0
m_textInfo.meshInfo[materialIndex].uvs0[0 + index_X4] = characterInfoArray[i].vertex_BL.uv;
m_textInfo.meshInfo[materialIndex].uvs0[1 + index_X4] = characterInfoArray[i].vertex_TL.uv;
m_textInfo.meshInfo[materialIndex].uvs0[2 + index_X4] = characterInfoArray[i].vertex_TR.uv;
m_textInfo.meshInfo[materialIndex].uvs0[3 + index_X4] = characterInfoArray[i].vertex_BR.uv;
if (isVolumetric)
{
m_textInfo.meshInfo[materialIndex].uvs0[4 + index_X4] = characterInfoArray[i].vertex_BL.uv;
m_textInfo.meshInfo[materialIndex].uvs0[5 + index_X4] = characterInfoArray[i].vertex_TL.uv;
m_textInfo.meshInfo[materialIndex].uvs0[6 + index_X4] = characterInfoArray[i].vertex_TR.uv;
m_textInfo.meshInfo[materialIndex].uvs0[7 + index_X4] = characterInfoArray[i].vertex_BR.uv;
}
// Setup UVS2
m_textInfo.meshInfo[materialIndex].uvs2[0 + index_X4] = characterInfoArray[i].vertex_BL.uv2;
m_textInfo.meshInfo[materialIndex].uvs2[1 + index_X4] = characterInfoArray[i].vertex_TL.uv2;
m_textInfo.meshInfo[materialIndex].uvs2[2 + index_X4] = characterInfoArray[i].vertex_TR.uv2;
m_textInfo.meshInfo[materialIndex].uvs2[3 + index_X4] = characterInfoArray[i].vertex_BR.uv2;
if (isVolumetric)
{
m_textInfo.meshInfo[materialIndex].uvs2[4 + index_X4] = characterInfoArray[i].vertex_BL.uv2;
m_textInfo.meshInfo[materialIndex].uvs2[5 + index_X4] = characterInfoArray[i].vertex_TL.uv2;
m_textInfo.meshInfo[materialIndex].uvs2[6 + index_X4] = characterInfoArray[i].vertex_TR.uv2;
m_textInfo.meshInfo[materialIndex].uvs2[7 + index_X4] = characterInfoArray[i].vertex_BR.uv2;
}
// Setup UVS4
//m_textInfo.meshInfo[0].uvs4[0 + index_X4] = characterInfoArray[i].vertex_BL.uv4;
//m_textInfo.meshInfo[0].uvs4[1 + index_X4] = characterInfoArray[i].vertex_TL.uv4;
//m_textInfo.meshInfo[0].uvs4[2 + index_X4] = characterInfoArray[i].vertex_TR.uv4;
//m_textInfo.meshInfo[0].uvs4[3 + index_X4] = characterInfoArray[i].vertex_BR.uv4;
// setup Vertex Colors
m_textInfo.meshInfo[materialIndex].colors32[0 + index_X4] = characterInfoArray[i].vertex_BL.color;
m_textInfo.meshInfo[materialIndex].colors32[1 + index_X4] = characterInfoArray[i].vertex_TL.color;
m_textInfo.meshInfo[materialIndex].colors32[2 + index_X4] = characterInfoArray[i].vertex_TR.color;
m_textInfo.meshInfo[materialIndex].colors32[3 + index_X4] = characterInfoArray[i].vertex_BR.color;
if (isVolumetric)
{
Color32 backColor = new Color32(255, 255, 128, 255);
m_textInfo.meshInfo[materialIndex].colors32[4 + index_X4] = backColor; //characterInfoArray[i].vertex_BL.color;
m_textInfo.meshInfo[materialIndex].colors32[5 + index_X4] = backColor; //characterInfoArray[i].vertex_TL.color;
m_textInfo.meshInfo[materialIndex].colors32[6 + index_X4] = backColor; //characterInfoArray[i].vertex_TR.color;
m_textInfo.meshInfo[materialIndex].colors32[7 + index_X4] = backColor; //characterInfoArray[i].vertex_BR.color;
}
m_textInfo.meshInfo[materialIndex].vertexCount = index_X4 + (!isVolumetric ? 4 : 8);
}
/// <summary>
/// Fill Vertex Buffers for Sprites
/// </summary>
/// <param name="i"></param>
/// <param name="spriteIndex_X4"></param>
protected virtual void FillSpriteVertexBuffers(int i, int index_X4)
{
int materialIndex = m_textInfo.characterInfo[i].materialReferenceIndex;
index_X4 = m_textInfo.meshInfo[materialIndex].vertexCount;
TMP_CharacterInfo[] characterInfoArray = m_textInfo.characterInfo;
m_textInfo.characterInfo[i].vertexIndex = index_X4;
// Setup Vertices for Characters
m_textInfo.meshInfo[materialIndex].vertices[0 + index_X4] = characterInfoArray[i].vertex_BL.position;
m_textInfo.meshInfo[materialIndex].vertices[1 + index_X4] = characterInfoArray[i].vertex_TL.position;
m_textInfo.meshInfo[materialIndex].vertices[2 + index_X4] = characterInfoArray[i].vertex_TR.position;
m_textInfo.meshInfo[materialIndex].vertices[3 + index_X4] = characterInfoArray[i].vertex_BR.position;
// Setup UVS0
m_textInfo.meshInfo[materialIndex].uvs0[0 + index_X4] = characterInfoArray[i].vertex_BL.uv;
m_textInfo.meshInfo[materialIndex].uvs0[1 + index_X4] = characterInfoArray[i].vertex_TL.uv;
m_textInfo.meshInfo[materialIndex].uvs0[2 + index_X4] = characterInfoArray[i].vertex_TR.uv;
m_textInfo.meshInfo[materialIndex].uvs0[3 + index_X4] = characterInfoArray[i].vertex_BR.uv;
// Setup UVS2
m_textInfo.meshInfo[materialIndex].uvs2[0 + index_X4] = characterInfoArray[i].vertex_BL.uv2;
m_textInfo.meshInfo[materialIndex].uvs2[1 + index_X4] = characterInfoArray[i].vertex_TL.uv2;
m_textInfo.meshInfo[materialIndex].uvs2[2 + index_X4] = characterInfoArray[i].vertex_TR.uv2;
m_textInfo.meshInfo[materialIndex].uvs2[3 + index_X4] = characterInfoArray[i].vertex_BR.uv2;
// Setup UVS4
//m_textInfo.meshInfo[0].uvs4[0 + index_X4] = characterInfoArray[i].vertex_BL.uv4;
//m_textInfo.meshInfo[0].uvs4[1 + index_X4] = characterInfoArray[i].vertex_TL.uv4;
//m_textInfo.meshInfo[0].uvs4[2 + index_X4] = characterInfoArray[i].vertex_TR.uv4;
//m_textInfo.meshInfo[0].uvs4[3 + index_X4] = characterInfoArray[i].vertex_BR.uv4;
// setup Vertex Colors
m_textInfo.meshInfo[materialIndex].colors32[0 + index_X4] = characterInfoArray[i].vertex_BL.color;
m_textInfo.meshInfo[materialIndex].colors32[1 + index_X4] = characterInfoArray[i].vertex_TL.color;
m_textInfo.meshInfo[materialIndex].colors32[2 + index_X4] = characterInfoArray[i].vertex_TR.color;
m_textInfo.meshInfo[materialIndex].colors32[3 + index_X4] = characterInfoArray[i].vertex_BR.color;
m_textInfo.meshInfo[materialIndex].vertexCount = index_X4 + 4;
}
/// <summary>
/// Method to add the underline geometry.
/// </summary>
/// <param name="start"></param>
/// <param name="end"></param>
/// <param name="startScale"></param>
/// <param name="endScale"></param>
/// <param name="maxScale"></param>
/// <param name="underlineColor"></param>
protected virtual void DrawUnderlineMesh(Vector3 start, Vector3 end, ref int index, float startScale, float endScale, float maxScale, float sdfScale, Color32 underlineColor)
{
if (m_cached_Underline_GlyphInfo == null)
{
if (!TMP_Settings.warningsDisabled) Debug.LogWarning("Unable to add underline since the Font Asset doesn't contain the underline character.", this);
return;
}
int verticesCount = index + 12;
// Check to make sure our current mesh buffer allocations can hold these new Quads.
if (verticesCount > m_textInfo.meshInfo[0].vertices.Length)
{
// Resize Mesh Buffers
m_textInfo.meshInfo[0].ResizeMeshInfo(verticesCount / 4);
}
// Adjust the position of the underline based on the lowest character. This matters for subscript character.
start.y = Mathf.Min(start.y, end.y);
end.y = Mathf.Min(start.y, end.y);
float segmentWidth = m_cached_Underline_GlyphInfo.width / 2 * maxScale;
if (end.x - start.x < m_cached_Underline_GlyphInfo.width * maxScale)
{
segmentWidth = (end.x - start.x) / 2f;
}
float startPadding = m_padding * startScale / maxScale;
float endPadding = m_padding * endScale / maxScale;
float underlineThickness = m_cached_Underline_GlyphInfo.height;
// UNDERLINE VERTICES FOR (3) LINE SEGMENTS
#region UNDERLINE VERTICES
Vector3[] vertices = m_textInfo.meshInfo[0].vertices;
// Front Part of the Underline
vertices[index + 0] = start + new Vector3(0, 0 - (underlineThickness + m_padding) * maxScale, 0); // BL
vertices[index + 1] = start + new Vector3(0, m_padding * maxScale, 0); // TL
vertices[index + 2] = vertices[index + 1] + new Vector3(segmentWidth, 0, 0); // TR
vertices[index + 3] = vertices[index + 0] + new Vector3(segmentWidth, 0, 0); // BR
// Middle Part of the Underline
vertices[index + 4] = vertices[index + 3]; // BL
vertices[index + 5] = vertices[index + 2]; // TL
vertices[index + 6] = end + new Vector3(-segmentWidth, m_padding * maxScale, 0); // TR
vertices[index + 7] = end + new Vector3(-segmentWidth, -(underlineThickness + m_padding) * maxScale, 0); // BR
// End Part of the Underline
vertices[index + 8] = vertices[index + 7]; // BL
vertices[index + 9] = vertices[index + 6]; // TL
vertices[index + 10] = end + new Vector3(0, m_padding * maxScale, 0); // TR
vertices[index + 11] = end + new Vector3(0, -(underlineThickness + m_padding) * maxScale, 0); // BR
#endregion
// UNDERLINE UV0
#region HANDLE UV0
Vector2[] uvs0 = m_textInfo.meshInfo[0].uvs0;
// Calculate UV required to setup the 3 Quads for the Underline.
Vector2 uv0 = new Vector2((m_cached_Underline_GlyphInfo.x - startPadding) / m_fontAsset.fontInfo.AtlasWidth, 1 - (m_cached_Underline_GlyphInfo.y + m_padding + m_cached_Underline_GlyphInfo.height) / m_fontAsset.fontInfo.AtlasHeight); // bottom left
Vector2 uv1 = new Vector2(uv0.x, 1 - (m_cached_Underline_GlyphInfo.y - m_padding) / m_fontAsset.fontInfo.AtlasHeight); // top left
Vector2 uv2 = new Vector2((m_cached_Underline_GlyphInfo.x - startPadding + m_cached_Underline_GlyphInfo.width / 2) / m_fontAsset.fontInfo.AtlasWidth, uv1.y); // Mid Top Left
Vector2 uv3 = new Vector2(uv2.x, uv0.y); // Mid Bottom Left
Vector2 uv4 = new Vector2((m_cached_Underline_GlyphInfo.x + endPadding + m_cached_Underline_GlyphInfo.width / 2) / m_fontAsset.fontInfo.AtlasWidth, uv1.y); // Mid Top Right
Vector2 uv5 = new Vector2(uv4.x, uv0.y); // Mid Bottom right
Vector2 uv6 = new Vector2((m_cached_Underline_GlyphInfo.x + endPadding + m_cached_Underline_GlyphInfo.width) / m_fontAsset.fontInfo.AtlasWidth, uv1.y); // End Part - Bottom Right
Vector2 uv7 = new Vector2(uv6.x, uv0.y); // End Part - Top Right
// Left Part of the Underline
uvs0[0 + index] = uv0; // BL
uvs0[1 + index] = uv1; // TL
uvs0[2 + index] = uv2; // TR
uvs0[3 + index] = uv3; // BR
// Middle Part of the Underline
uvs0[4 + index] = new Vector2(uv2.x - uv2.x * 0.001f, uv0.y);
uvs0[5 + index] = new Vector2(uv2.x - uv2.x * 0.001f, uv1.y);
uvs0[6 + index] = new Vector2(uv2.x + uv2.x * 0.001f, uv1.y);
uvs0[7 + index] = new Vector2(uv2.x + uv2.x * 0.001f, uv0.y);
// Right Part of the Underline
uvs0[8 + index] = uv5;
uvs0[9 + index] = uv4;
uvs0[10 + index] = uv6;
uvs0[11 + index] = uv7;
#endregion
// UNDERLINE UV2
#region HANDLE UV2 - SDF SCALE
// UV1 contains Face / Border UV layout.
float min_UvX = 0;
float max_UvX = (vertices[index + 2].x - start.x) / (end.x - start.x);
//Calculate the xScale or how much the UV's are getting stretched on the X axis for the middle section of the underline.
float xScale = Mathf.Abs(sdfScale);
Vector2[] uvs2 = m_textInfo.meshInfo[0].uvs2;
uvs2[0 + index] = PackUV(0, 0, xScale);
uvs2[1 + index] = PackUV(0, 1, xScale);
uvs2[2 + index] = PackUV(max_UvX, 1, xScale);
uvs2[3 + index] = PackUV(max_UvX, 0, xScale);
min_UvX = (vertices[index + 4].x - start.x) / (end.x - start.x);
max_UvX = (vertices[index + 6].x - start.x) / (end.x - start.x);
uvs2[4 + index] = PackUV(min_UvX, 0, xScale);
uvs2[5 + index] = PackUV(min_UvX, 1, xScale);
uvs2[6 + index] = PackUV(max_UvX, 1, xScale);
uvs2[7 + index] = PackUV(max_UvX, 0, xScale);
min_UvX = (vertices[index + 8].x - start.x) / (end.x - start.x);
max_UvX = (vertices[index + 6].x - start.x) / (end.x - start.x);
uvs2[8 + index] = PackUV(min_UvX, 0, xScale);
uvs2[9 + index] = PackUV(min_UvX, 1, xScale);
uvs2[10 + index] = PackUV(1, 1, xScale);
uvs2[11 + index] = PackUV(1, 0, xScale);
#endregion
// UNDERLINE VERTEX COLORS
#region
// Alpha is the lower of the vertex color or tag color alpha used.
underlineColor.a = m_fontColor32.a < underlineColor.a ? (byte)(m_fontColor32.a) : (byte)(underlineColor.a);
Color32[] colors32 = m_textInfo.meshInfo[0].colors32;
colors32[0 + index] = underlineColor;
colors32[1 + index] = underlineColor;
colors32[2 + index] = underlineColor;
colors32[3 + index] = underlineColor;
colors32[4 + index] = underlineColor;
colors32[5 + index] = underlineColor;
colors32[6 + index] = underlineColor;
colors32[7 + index] = underlineColor;
colors32[8 + index] = underlineColor;
colors32[9 + index] = underlineColor;
colors32[10 + index] = underlineColor;
colors32[11 + index] = underlineColor;
#endregion
index += 12;
}
protected virtual void DrawTextHighlight(Vector3 start, Vector3 end, ref int index, Color32 highlightColor)
{
if (m_cached_Underline_GlyphInfo == null)
{
if (!TMP_Settings.warningsDisabled) Debug.LogWarning("Unable to add underline since the Font Asset doesn't contain the underline character.", this);
return;
}
int verticesCount = index + 4;
// Check to make sure our current mesh buffer allocations can hold these new Quads.
if (verticesCount > m_textInfo.meshInfo[0].vertices.Length)
{
// Resize Mesh Buffers
m_textInfo.meshInfo[0].ResizeMeshInfo(verticesCount / 4);
}
// UNDERLINE VERTICES FOR (3) LINE SEGMENTS
#region HIGHLIGHT VERTICES
Vector3[] vertices = m_textInfo.meshInfo[0].vertices;
// Front Part of the Underline
vertices[index + 0] = start; // BL
vertices[index + 1] = new Vector3(start.x, end.y, 0); // TL
vertices[index + 2] = end; // TR
vertices[index + 3] = new Vector3(end.x, start.y, 0); // BR
#endregion
// UNDERLINE UV0
#region HANDLE UV0
Vector2[] uvs0 = m_textInfo.meshInfo[0].uvs0;
// Calculate UV required to setup the 3 Quads for the Underline.
Vector2 uv0 = new Vector2((m_cached_Underline_GlyphInfo.x + m_cached_Underline_GlyphInfo.width / 2) / m_fontAsset.fontInfo.AtlasWidth, 1 - (m_cached_Underline_GlyphInfo.y + m_cached_Underline_GlyphInfo.height / 2) / m_fontAsset.fontInfo.AtlasHeight); // bottom left
//Vector2 uv1 = new Vector2(uv0.x, uv0.y); // top left
//Vector2 uv2 = new Vector2(uv0.x, uv0.y); // Top Right
//Vector2 uv3 = new Vector2(uv2.x, uv0.y); // Bottom Right
// Left Part of the Underline
uvs0[0 + index] = uv0; // BL
uvs0[1 + index] = uv0; // TL
uvs0[2 + index] = uv0; // TR
uvs0[3 + index] = uv0; // BR
#endregion
// UNDERLINE UV2
#region HANDLE UV2 - SDF SCALE
// UV1 contains Face / Border UV layout.
//float min_UvX = 0;
//float max_UvX = (vertices[index + 2].x - start.x) / (end.x - start.x);
////Calculate the xScale or how much the UV's are getting stretched on the X axis for the middle section of the underline.
//float xScale = 0; // Mathf.Abs(sdfScale);
Vector2[] uvs2 = m_textInfo.meshInfo[0].uvs2;
Vector2 customUV = new Vector2(0, 1);
uvs2[0 + index] = customUV; // PackUV(-0.2f, -0.2f, xScale);
uvs2[1 + index] = customUV; // PackUV(-0.2f, -0.1f, xScale);
uvs2[2 + index] = customUV; // PackUV(-0.1f, -0.1f, xScale);
uvs2[3 + index] = customUV; // PackUV(-0.1f, -0.2f, xScale);
#endregion
// HIGHLIGHT VERTEX COLORS
#region
// Alpha is the lower of the vertex color or tag color alpha used.
highlightColor.a = m_fontColor32.a < highlightColor.a ? m_fontColor32.a : highlightColor.a;
Color32[] colors32 = m_textInfo.meshInfo[0].colors32;
colors32[0 + index] = highlightColor;
colors32[1 + index] = highlightColor;
colors32[2 + index] = highlightColor;
colors32[3 + index] = highlightColor;
#endregion
index += 4;
}
/// <summary>
/// Internal function used to load the default settings of text objects.
/// </summary>
protected void LoadDefaultSettings()
{
if (m_text == null || m_isWaitingOnResourceLoad)
{
if (TMP_Settings.autoSizeTextContainer)
autoSizeTextContainer = true;
else
{
m_rectTransform = this.rectTransform;
if (GetType() == typeof(TextMeshPro))
m_rectTransform.sizeDelta = TMP_Settings.defaultTextMeshProTextContainerSize;
else
m_rectTransform.sizeDelta = TMP_Settings.defaultTextMeshProUITextContainerSize;
}
m_enableWordWrapping = TMP_Settings.enableWordWrapping;
m_enableKerning = TMP_Settings.enableKerning;
m_enableExtraPadding = TMP_Settings.enableExtraPadding;
m_tintAllSprites = TMP_Settings.enableTintAllSprites;
m_parseCtrlCharacters = TMP_Settings.enableParseEscapeCharacters;
m_fontSize = m_fontSizeBase = TMP_Settings.defaultFontSize;
m_fontSizeMin = m_fontSize * TMP_Settings.defaultTextAutoSizingMinRatio;
m_fontSizeMax = m_fontSize * TMP_Settings.defaultTextAutoSizingMaxRatio;
m_isAlignmentEnumConverted = true;
m_isWaitingOnResourceLoad = false;
}
else if (m_isAlignmentEnumConverted == false)
{
// Convert TextAlignmentOptions enumerations.
m_isAlignmentEnumConverted = true;
m_textAlignment = TMP_Compatibility.ConvertTextAlignmentEnumValues(m_textAlignment);
}
}
/// <summary>
/// Method used to find and cache references to the Underline and Ellipsis characters.
/// </summary>
/// <param name=""></param>
protected void GetSpecialCharacters(TMP_FontAsset fontAsset)
{
// Check & Assign Underline Character for use with the Underline tag.
if (!fontAsset.characterDictionary.TryGetValue(95, out m_cached_Underline_GlyphInfo)) //95
{
// Check fallback fonts
// TODO
}
// Check & Assign Underline Character for use with the Underline tag.
if (!fontAsset.characterDictionary.TryGetValue(8230, out m_cached_Ellipsis_GlyphInfo)) //95
{
// Check fallback fonts
// TODO
}
}
/// <summary>
/// Replace a given number of characters (tag) in the array with a new character and shift subsequent characters in the array.
/// </summary>
/// <param name="chars">Array which contains the text.</param>
/// <param name="insertionIndex">The index of where the new character will be inserted</param>
/// <param name="tagLength">Length of the tag being replaced.</param>
/// <param name="c">The replacement character.</param>
protected void ReplaceTagWithCharacter(int[] chars, int insertionIndex, int tagLength, char c)
{
chars[insertionIndex] = c;
for (int i = insertionIndex + tagLength; i < chars.Length; i++)
{
chars[i - 3] = chars[i];
}
}
/// <summary>
///
/// </summary>
/// <returns></returns>
//protected int GetMaterialReferenceForFontWeight()
//{
// //bool isItalic = (m_style & FontStyles.Italic) == FontStyles.Italic || (m_fontStyle & FontStyles.Italic) == FontStyles.Italic;
// m_currentMaterialIndex = MaterialReference.AddMaterialReference(m_currentFontAsset.fontWeights[0].italicTypeface.material, m_currentFontAsset.fontWeights[0].italicTypeface, m_materialReferences, m_materialReferenceIndexLookup);
// return 0;
//}
/// <summary>
///
/// </summary>
/// <returns></returns>
protected TMP_FontAsset GetFontAssetForWeight(int fontWeight)
{
bool isItalic = (m_style & FontStyles.Italic) == FontStyles.Italic || (m_fontStyle & FontStyles.Italic) == FontStyles.Italic;
TMP_FontAsset fontAsset = null;
int weightIndex = fontWeight / 100;
if (isItalic)
fontAsset = m_currentFontAsset.fontWeights[weightIndex].italicTypeface;
else
fontAsset = m_currentFontAsset.fontWeights[weightIndex].regularTypeface;
return fontAsset;
}
/// <summary>
/// Method to Enable or Disable child SubMesh objects.
/// </summary>
/// <param name="state"></param>
protected virtual void SetActiveSubMeshes(bool state) { }
/// <summary>
/// Destroy Sub Mesh Objects.
/// </summary>
protected virtual void ClearSubMeshObjects() { }
/// <summary>
/// Function to clear the geometry of the Primary and Sub Text objects.
/// </summary>
public virtual void ClearMesh() { }
/// <summary>
/// Function to clear the geometry of the Primary and Sub Text objects.
/// </summary>
public virtual void ClearMesh(bool uploadGeometry) { }
/// <summary>
/// Function which returns the text after it has been parsed and rich text tags removed.
/// </summary>
/// <returns></returns>
public virtual string GetParsedText()
{
if (m_textInfo == null)
return string.Empty;
int characterCount = m_textInfo.characterCount;
// TODO - Could implement some static buffer pool shared by all instances of TMP objects.
char[] buffer = new char[characterCount];
for (int i = 0; i < characterCount && i < m_textInfo.characterInfo.Length; i++)
{
buffer[i] = m_textInfo.characterInfo[i].character;
}
return new string(buffer);
}
/// <summary>
/// Function to pack scale information in the UV2 Channel.
/// </summary>
/// <param name="x"></param>
/// <param name="y"></param>
/// <param name="scale"></param>
/// <returns></returns>
//protected Vector2 PackUV(float x, float y, float scale)
//{
// Vector2 output;
// output.x = Mathf.Floor(x * 4095);
// output.y = Mathf.Floor(y * 4095);
// output.x = (output.x * 4096) + output.y;
// output.y = scale;
// return output;
//}
/// <summary>
/// Function to pack scale information in the UV2 Channel.
/// </summary>
/// <param name="x"></param>
/// <param name="y"></param>
/// <param name="scale"></param>
/// <returns></returns>
protected Vector2 PackUV(float x, float y, float scale)
{
Vector2 output;
output.x = (int)(x * 511);
output.y = (int)(y * 511);
output.x = (output.x * 4096) + output.y;
output.y = scale;
return output;
}
/// <summary>
///
/// </summary>
/// <param name="x"></param>
/// <param name="y"></param>
/// <returns></returns>
protected float PackUV(float x, float y)
{
double x0 = (int)(x * 511);
double y0 = (int)(y * 511);
return (float)((x0 * 4096) + y0);
}
/// <summary>
/// Function to pack scale information in the UV2 Channel.
/// </summary>
/// <param name="x"></param>
/// <param name="y"></param>
/// <param name="scale"></param>
/// <returns></returns>
//protected Vector2 PackUV(float x, float y, float scale)
//{
// Vector2 output;
// output.x = Mathf.Floor(x * 4095);
// output.y = Mathf.Floor(y * 4095);
// return new Vector2((output.x * 4096) + output.y, scale);
//}
/// <summary>
///
/// </summary>
/// <param name="x"></param>
/// <param name="y"></param>
/// <returns></returns>
//protected float PackUV(float x, float y)
//{
// x = (x % 5) / 5;
// y = (y % 5) / 5;
// return Mathf.Round(x * 4096) + y;
//}
/// <summary>
/// Method to convert Hex to Int
/// </summary>
/// <param name="hex"></param>
/// <returns></returns>
protected int HexToInt(char hex)
{
switch (hex)
{
case '0': return 0;
case '1': return 1;
case '2': return 2;
case '3': return 3;
case '4': return 4;
case '5': return 5;
case '6': return 6;
case '7': return 7;
case '8': return 8;
case '9': return 9;
case 'A': return 10;
case 'B': return 11;
case 'C': return 12;
case 'D': return 13;
case 'E': return 14;
case 'F': return 15;
case 'a': return 10;
case 'b': return 11;
case 'c': return 12;
case 'd': return 13;
case 'e': return 14;
case 'f': return 15;
}
return 15;
}
/// <summary>
/// Convert UTF-16 Hex to Char
/// </summary>
/// <returns>The Unicode hex.</returns>
/// <param name="i">The index.</param>
protected int GetUTF16(string text, int i)
{
int unicode = 0;
unicode += HexToInt(text[i]) << 12;
unicode += HexToInt(text[i + 1]) << 8;
unicode += HexToInt(text[i + 2]) << 4;
unicode += HexToInt(text[i + 3]);
return unicode;
}
/// <summary>
/// Convert UTF-16 Hex to Char
/// </summary>
/// <returns>The Unicode hex.</returns>
/// <param name="i">The index.</param>
protected int GetUTF16(StringBuilder text, int i)
{
int unicode = 0;
unicode += HexToInt(text[i]) << 12;
unicode += HexToInt(text[i + 1]) << 8;
unicode += HexToInt(text[i + 2]) << 4;
unicode += HexToInt(text[i + 3]);
return unicode;
}
/// <summary>
/// Convert UTF-32 Hex to Char
/// </summary>
/// <returns>The Unicode hex.</returns>
/// <param name="i">The index.</param>
protected int GetUTF32(string text, int i)
{
int unicode = 0;
unicode += HexToInt(text[i]) << 30;
unicode += HexToInt(text[i + 1]) << 24;
unicode += HexToInt(text[i + 2]) << 20;
unicode += HexToInt(text[i + 3]) << 16;
unicode += HexToInt(text[i + 4]) << 12;
unicode += HexToInt(text[i + 5]) << 8;
unicode += HexToInt(text[i + 6]) << 4;
unicode += HexToInt(text[i + 7]);
return unicode;
}
/// <summary>
/// Convert UTF-32 Hex to Char
/// </summary>
/// <returns>The Unicode hex.</returns>
/// <param name="i">The index.</param>
protected int GetUTF32(StringBuilder text, int i)
{
int unicode = 0;
unicode += HexToInt(text[i]) << 30;
unicode += HexToInt(text[i + 1]) << 24;
unicode += HexToInt(text[i + 2]) << 20;
unicode += HexToInt(text[i + 3]) << 16;
unicode += HexToInt(text[i + 4]) << 12;
unicode += HexToInt(text[i + 5]) << 8;
unicode += HexToInt(text[i + 6]) << 4;
unicode += HexToInt(text[i + 7]);
return unicode;
}
/// <summary>
/// Method to convert Hex color values to Color32
/// </summary>
/// <param name="hexChars"></param>
/// <param name="tagCount"></param>
/// <returns></returns>
protected Color32 HexCharsToColor(char[] hexChars, int tagCount)
{
if (tagCount == 4)
{
byte r = (byte)(HexToInt(hexChars[1]) * 16 + HexToInt(hexChars[1]));
byte g = (byte)(HexToInt(hexChars[2]) * 16 + HexToInt(hexChars[2]));
byte b = (byte)(HexToInt(hexChars[3]) * 16 + HexToInt(hexChars[3]));
return new Color32(r, g, b, 255);
}
else if (tagCount == 5)
{
byte r = (byte)(HexToInt(hexChars[1]) * 16 + HexToInt(hexChars[1]));
byte g = (byte)(HexToInt(hexChars[2]) * 16 + HexToInt(hexChars[2]));
byte b = (byte)(HexToInt(hexChars[3]) * 16 + HexToInt(hexChars[3]));
byte a = (byte)(HexToInt(hexChars[4]) * 16 + HexToInt(hexChars[4]));
return new Color32(r, g, b, a);
}
else if (tagCount == 7)
{
byte r = (byte)(HexToInt(hexChars[1]) * 16 + HexToInt(hexChars[2]));
byte g = (byte)(HexToInt(hexChars[3]) * 16 + HexToInt(hexChars[4]));
byte b = (byte)(HexToInt(hexChars[5]) * 16 + HexToInt(hexChars[6]));
return new Color32(r, g, b, 255);
}
else if (tagCount == 9)
{
byte r = (byte)(HexToInt(hexChars[1]) * 16 + HexToInt(hexChars[2]));
byte g = (byte)(HexToInt(hexChars[3]) * 16 + HexToInt(hexChars[4]));
byte b = (byte)(HexToInt(hexChars[5]) * 16 + HexToInt(hexChars[6]));
byte a = (byte)(HexToInt(hexChars[7]) * 16 + HexToInt(hexChars[8]));
return new Color32(r, g, b, a);
}
else if (tagCount == 10)
{
byte r = (byte)(HexToInt(hexChars[7]) * 16 + HexToInt(hexChars[7]));
byte g = (byte)(HexToInt(hexChars[8]) * 16 + HexToInt(hexChars[8]));
byte b = (byte)(HexToInt(hexChars[9]) * 16 + HexToInt(hexChars[9]));
return new Color32(r, g, b, 255);
}
else if (tagCount == 11)
{
byte r = (byte)(HexToInt(hexChars[7]) * 16 + HexToInt(hexChars[7]));
byte g = (byte)(HexToInt(hexChars[8]) * 16 + HexToInt(hexChars[8]));
byte b = (byte)(HexToInt(hexChars[9]) * 16 + HexToInt(hexChars[9]));
byte a = (byte)(HexToInt(hexChars[10]) * 16 + HexToInt(hexChars[10]));
return new Color32(r, g, b, a);
}
else if (tagCount == 13)
{
byte r = (byte)(HexToInt(hexChars[7]) * 16 + HexToInt(hexChars[8]));
byte g = (byte)(HexToInt(hexChars[9]) * 16 + HexToInt(hexChars[10]));
byte b = (byte)(HexToInt(hexChars[11]) * 16 + HexToInt(hexChars[12]));
return new Color32(r, g, b, 255);
}
else if (tagCount == 15)
{
byte r = (byte)(HexToInt(hexChars[7]) * 16 + HexToInt(hexChars[8]));
byte g = (byte)(HexToInt(hexChars[9]) * 16 + HexToInt(hexChars[10]));
byte b = (byte)(HexToInt(hexChars[11]) * 16 + HexToInt(hexChars[12]));
byte a = (byte)(HexToInt(hexChars[13]) * 16 + HexToInt(hexChars[14]));
return new Color32(r, g, b, a);
}
return new Color32(255, 255, 255, 255);
}
/// <summary>
/// Method to convert Hex Color values to Color32
/// </summary>
/// <param name="hexChars"></param>
/// <param name="startIndex"></param>
/// <param name="length"></param>
/// <returns></returns>
protected Color32 HexCharsToColor(char[] hexChars, int startIndex, int length)
{
if (length == 7)
{
byte r = (byte)(HexToInt(hexChars[startIndex + 1]) * 16 + HexToInt(hexChars[startIndex + 2]));
byte g = (byte)(HexToInt(hexChars[startIndex + 3]) * 16 + HexToInt(hexChars[startIndex + 4]));
byte b = (byte)(HexToInt(hexChars[startIndex + 5]) * 16 + HexToInt(hexChars[startIndex + 6]));
return new Color32(r, g, b, 255);
}
else if (length == 9)
{
byte r = (byte)(HexToInt(hexChars[startIndex + 1]) * 16 + HexToInt(hexChars[startIndex + 2]));
byte g = (byte)(HexToInt(hexChars[startIndex + 3]) * 16 + HexToInt(hexChars[startIndex + 4]));
byte b = (byte)(HexToInt(hexChars[startIndex + 5]) * 16 + HexToInt(hexChars[startIndex + 6]));
byte a = (byte)(HexToInt(hexChars[startIndex + 7]) * 16 + HexToInt(hexChars[startIndex + 8]));
return new Color32(r, g, b, a);
}
return s_colorWhite;
}
/// <summary>
/// Method which returns the number of parameters used in a tag attribute and populates an array with such values.
/// </summary>
/// <param name="chars">Char[] containing the tag attribute and data</param>
/// <param name="startIndex">The index of the first char of the data</param>
/// <param name="length">The length of the data</param>
/// <param name="parameters">The number of parameters contained in the Char[]</param>
/// <returns></returns>
int GetAttributeParameters(char[] chars, int startIndex, int length, ref float[] parameters)
{
int endIndex = startIndex;
int attributeCount = 0;
while (endIndex < startIndex + length)
{
parameters[attributeCount] = ConvertToFloat(chars, startIndex, length, out endIndex);
length -= (endIndex - startIndex) + 1;
startIndex = endIndex + 1;
attributeCount += 1;
}
return attributeCount;
}
/// <summary>
/// Extracts a float value from char[] assuming we know the position of the start, end and decimal point.
/// </summary>
/// <param name="chars"></param>
/// <param name="startIndex"></param>
/// <param name="length"></param>
/// <returns></returns>
protected float ConvertToFloat(char[] chars, int startIndex, int length)
{
int lastIndex = 0;
return ConvertToFloat(chars, startIndex, length, out lastIndex);
}
/// <summary>
/// Extracts a float value from char[] given a start index and length.
/// </summary>
/// <param name="chars"></param> The Char[] containing the numerical sequence.
/// <param name="startIndex"></param> The index of the start of the numerical sequence.
/// <param name="length"></param> The length of the numerical sequence.
/// <param name="lastIndex"></param> Index of the last character in the validated sequence.
/// <returns></returns>
protected float ConvertToFloat(char[] chars, int startIndex, int length, out int lastIndex)
{
if (startIndex == 0) { lastIndex = 0; return -9999; }
int endIndex = startIndex + length;
bool isIntegerValue = true;
float decimalPointMultiplier = 0;
// Set value multiplier checking the first character to determine if we are using '+' or '-'
int valueSignMultiplier = 1;
if (chars[startIndex] == '+')
{
valueSignMultiplier = 1;
startIndex += 1;
}
else if (chars[startIndex] == '-')
{
valueSignMultiplier = -1;
startIndex += 1;
}
float value = 0;
for (int i = startIndex; i < endIndex; i++)
{
uint c = chars[i];
if (c >= '0' && c <= '9' || c == '.')
{
if (c == '.')
{
isIntegerValue = false;
decimalPointMultiplier = 0.1f;
continue;
}
//Calculate integer and floating point value
if (isIntegerValue)
value = value * 10 + (c - 48) * valueSignMultiplier;
else
{
value = value + (c - 48) * decimalPointMultiplier * valueSignMultiplier;
decimalPointMultiplier *= 0.1f;
}
continue;
}
else if (c == ',')
{
if (i + 1 < endIndex && chars[i + 1] == ' ')
lastIndex = i + 1;
else
lastIndex = i;
return value;
}
}
lastIndex = endIndex;
return value;
}
/// <summary>
/// Function to identify and validate the rich tag. Returns the position of the > if the tag was valid.
/// </summary>
/// <param name="chars"></param>
/// <param name="startIndex"></param>
/// <param name="endIndex"></param>
/// <returns></returns>
protected bool ValidateHtmlTag(int[] chars, int startIndex, out int endIndex)
{
int tagCharCount = 0;
byte attributeFlag = 0;
TagUnits tagUnits = TagUnits.Pixels;
TagType tagType = TagType.None;
int attributeIndex = 0;
m_xmlAttribute[attributeIndex].nameHashCode = 0;
m_xmlAttribute[attributeIndex].valueType = TagType.None;
m_xmlAttribute[attributeIndex].valueHashCode = 0;
m_xmlAttribute[attributeIndex].valueStartIndex = 0;
m_xmlAttribute[attributeIndex].valueLength = 0;
// Clear attribute name hash codes
m_xmlAttribute[1].nameHashCode = 0;
m_xmlAttribute[2].nameHashCode = 0;
m_xmlAttribute[3].nameHashCode = 0;
m_xmlAttribute[4].nameHashCode = 0;
endIndex = startIndex;
bool isTagSet = false;
bool isValidHtmlTag = false;
for (int i = startIndex; i < chars.Length && chars[i] != 0 && tagCharCount < m_htmlTag.Length && chars[i] != 60; i++)
{
if (chars[i] == 62) // ASCII Code of End HTML tag '>'
{
isValidHtmlTag = true;
endIndex = i;
m_htmlTag[tagCharCount] = (char)0;
break;
}
m_htmlTag[tagCharCount] = (char)chars[i];
tagCharCount += 1;
if (attributeFlag == 1)
{
if (tagType == TagType.None)
{
// Check for attribute type
if (chars[i] == 43 || chars[i] == 45 || chars[i] == 46 || (chars[i] >= '0' && chars[i] <= '9'))
{
tagType = TagType.NumericalValue;
m_xmlAttribute[attributeIndex].valueType = TagType.NumericalValue;
m_xmlAttribute[attributeIndex].valueStartIndex = tagCharCount - 1;
m_xmlAttribute[attributeIndex].valueLength += 1;
}
else if (chars[i] == 35)
{
tagType = TagType.ColorValue;
m_xmlAttribute[attributeIndex].valueType = TagType.ColorValue;
m_xmlAttribute[attributeIndex].valueStartIndex = tagCharCount - 1;
m_xmlAttribute[attributeIndex].valueLength += 1;
}
else if (chars[i] == 34)
{
tagType = TagType.StringValue;
m_xmlAttribute[attributeIndex].valueType = TagType.StringValue;
m_xmlAttribute[attributeIndex].valueStartIndex = tagCharCount;
}
else
{
tagType = TagType.StringValue;
m_xmlAttribute[attributeIndex].valueType = TagType.StringValue;
m_xmlAttribute[attributeIndex].valueStartIndex = tagCharCount - 1;
m_xmlAttribute[attributeIndex].valueHashCode = (m_xmlAttribute[attributeIndex].valueHashCode << 5) + m_xmlAttribute[attributeIndex].valueHashCode ^ chars[i];
m_xmlAttribute[attributeIndex].valueLength += 1;
}
}
else
{
if (tagType == TagType.NumericalValue)
{
// Check for termination of numerical value.
if (chars[i] == 112 || chars[i] == 101 || chars[i] == 37 || chars[i] == 32)
{
attributeFlag = 2;
tagType = TagType.None;
attributeIndex += 1;
m_xmlAttribute[attributeIndex].nameHashCode = 0;
m_xmlAttribute[attributeIndex].valueType = TagType.None;
m_xmlAttribute[attributeIndex].valueHashCode = 0;
m_xmlAttribute[attributeIndex].valueStartIndex = 0;
m_xmlAttribute[attributeIndex].valueLength = 0;
if (chars[i] == 101)
tagUnits = TagUnits.FontUnits;
else if (chars[i] == 37)
tagUnits = TagUnits.Percentage;
}
else if (attributeFlag != 2)
{
m_xmlAttribute[attributeIndex].valueLength += 1;
}
}
else if (tagType == TagType.ColorValue)
{
if (chars[i] != 32)
{
m_xmlAttribute[attributeIndex].valueLength += 1;
}
else
{
attributeFlag = 2;
tagType = TagType.None;
attributeIndex += 1;
m_xmlAttribute[attributeIndex].nameHashCode = 0;
m_xmlAttribute[attributeIndex].valueType = TagType.None;
m_xmlAttribute[attributeIndex].valueHashCode = 0;
m_xmlAttribute[attributeIndex].valueStartIndex = 0;
m_xmlAttribute[attributeIndex].valueLength = 0;
}
}
else if (tagType == TagType.StringValue)
{
// Compute HashCode value for the named tag.
if (chars[i] != 34)
{
m_xmlAttribute[attributeIndex].valueHashCode = (m_xmlAttribute[attributeIndex].valueHashCode << 5) + m_xmlAttribute[attributeIndex].valueHashCode ^ chars[i];
m_xmlAttribute[attributeIndex].valueLength += 1;
}
else
{
//m_xmlAttribute[attributeIndex].valueHashCode = -1;
attributeFlag = 2;
tagType = TagType.None;
attributeIndex += 1;
m_xmlAttribute[attributeIndex].nameHashCode = 0;
m_xmlAttribute[attributeIndex].valueType = TagType.None;
m_xmlAttribute[attributeIndex].valueHashCode = 0;
m_xmlAttribute[attributeIndex].valueStartIndex = 0;
m_xmlAttribute[attributeIndex].valueLength = 0;
}
}
}
}
if (chars[i] == 61) // '='
attributeFlag = 1;
// Compute HashCode for the name of the attribute
if (attributeFlag == 0 && chars[i] == 32)
{
if (isTagSet) return false;
isTagSet = true;
attributeFlag = 2;
tagType = TagType.None;
attributeIndex += 1;
m_xmlAttribute[attributeIndex].nameHashCode = 0;
m_xmlAttribute[attributeIndex].valueType = TagType.None;
m_xmlAttribute[attributeIndex].valueHashCode = 0;
m_xmlAttribute[attributeIndex].valueStartIndex = 0;
m_xmlAttribute[attributeIndex].valueLength = 0;
}
if (attributeFlag == 0)
m_xmlAttribute[attributeIndex].nameHashCode = (m_xmlAttribute[attributeIndex].nameHashCode << 3) - m_xmlAttribute[attributeIndex].nameHashCode + chars[i];
if (attributeFlag == 2 && chars[i] == 32)
attributeFlag = 0;
}
if (!isValidHtmlTag)
{
return false;
}
//Debug.Log("Tag is [" + m_htmlTag.ArrayToString() + "]. Tag HashCode: " + m_xmlAttribute[0].nameHashCode + " Tag Value HashCode: " + m_xmlAttribute[0].valueHashCode + " Attribute 1 HashCode: " + m_xmlAttribute[1].nameHashCode + " Value HashCode: " + m_xmlAttribute[1].valueHashCode);
//for (int i = 0; i < attributeIndex + 1; i++)
// Debug.Log("Tag [" + i + "] with HashCode: " + m_xmlAttribute[i].nameHashCode + " has value of [" + new string(m_htmlTag, m_xmlAttribute[i].valueStartIndex, m_xmlAttribute[i].valueLength) + "] Numerical Value: " + ConvertToFloat(m_htmlTag, m_xmlAttribute[i].valueStartIndex, m_xmlAttribute[i].valueLength));
// Special handling of the no parsing tag </noparse> </NOPARSE> tag
if (tag_NoParsing && (m_xmlAttribute[0].nameHashCode != 53822163 && m_xmlAttribute[0].nameHashCode != 49429939))
return false;
else if (m_xmlAttribute[0].nameHashCode == 53822163 || m_xmlAttribute[0].nameHashCode == 49429939)
{
tag_NoParsing = false;
return true;
}
// Color <#FFF> 3 Hex values (short form)
if (m_htmlTag[0] == 35 && tagCharCount == 4)
{
m_htmlColor = HexCharsToColor(m_htmlTag, tagCharCount);
m_colorStack.Add(m_htmlColor);
return true;
}
// Color <#FFF7> 4 Hex values with alpha (short form)
else if (m_htmlTag[0] == 35 && tagCharCount == 5)
{
m_htmlColor = HexCharsToColor(m_htmlTag, tagCharCount);
m_colorStack.Add(m_htmlColor);
return true;
}
// Color <#FF00FF>
else if (m_htmlTag[0] == 35 && tagCharCount == 7) // if Tag begins with # and contains 7 characters.
{
m_htmlColor = HexCharsToColor(m_htmlTag, tagCharCount);
m_colorStack.Add(m_htmlColor);
return true;
}
// Color <#FF00FF00> with alpha
else if (m_htmlTag[0] == 35 && tagCharCount == 9) // if Tag begins with # and contains 9 characters.
{
m_htmlColor = HexCharsToColor(m_htmlTag, tagCharCount);
m_colorStack.Add(m_htmlColor);
return true;
}
else
{
float value = 0;
switch (m_xmlAttribute[0].nameHashCode)
{
case 98: // <b>
case 66: // <B>
m_style |= FontStyles.Bold;
m_fontStyleStack.Add(FontStyles.Bold);
m_fontWeightInternal = 700;
m_fontWeightStack.Add(700);
return true;
case 427: // </b>
case 395: // </B>
if ((m_fontStyle & FontStyles.Bold) != FontStyles.Bold)
{
m_fontWeightInternal = m_fontWeightStack.Remove();
if (m_fontStyleStack.Remove(FontStyles.Bold) == 0)
m_style &= ~FontStyles.Bold;
}
return true;
case 105: // <i>
case 73: // <I>
m_style |= FontStyles.Italic;
m_fontStyleStack.Add(FontStyles.Italic);
return true;
case 434: // </i>
case 402: // </I>
if (m_fontStyleStack.Remove(FontStyles.Italic) == 0)
m_style &= ~FontStyles.Italic;
return true;
case 115: // <s>
case 83: // <S>
m_style |= FontStyles.Strikethrough;
m_fontStyleStack.Add(FontStyles.Strikethrough);
if (m_xmlAttribute[1].nameHashCode == 281955 || m_xmlAttribute[1].nameHashCode == 192323)
{
m_strikethroughColor = HexCharsToColor(m_htmlTag, m_xmlAttribute[1].valueStartIndex, m_xmlAttribute[1].valueLength);
m_strikethroughColor.a = m_htmlColor.a < m_strikethroughColor.a ? (byte)(m_htmlColor.a) : (byte)(m_strikethroughColor .a);
}
else
m_strikethroughColor = m_htmlColor;
m_strikethroughColorStack.Add(m_strikethroughColor);
return true;
case 444: // </s>
case 412: // </S>
if ((m_fontStyle & FontStyles.Strikethrough) != FontStyles.Strikethrough)
{
if (m_fontStyleStack.Remove(FontStyles.Strikethrough) == 0)
m_style &= ~FontStyles.Strikethrough;
}
return true;
case 117: // <u>
case 85: // <U>
m_style |= FontStyles.Underline;
m_fontStyleStack.Add(FontStyles.Underline);
if (m_xmlAttribute[1].nameHashCode == 281955 || m_xmlAttribute[1].nameHashCode == 192323)
{
m_underlineColor = HexCharsToColor(m_htmlTag, m_xmlAttribute[1].valueStartIndex, m_xmlAttribute[1].valueLength);
m_underlineColor.a = m_htmlColor.a < m_underlineColor.a ? (byte)(m_htmlColor.a) : (byte)(m_underlineColor.a);
}
else
m_underlineColor = m_htmlColor;
m_underlineColorStack.Add(m_underlineColor);
return true;
case 446: // </u>
case 414: // </U>
if ((m_fontStyle & FontStyles.Underline) != FontStyles.Underline)
{
m_underlineColor = m_underlineColorStack.Remove();
if (m_fontStyleStack.Remove(FontStyles.Underline) == 0)
m_style &= ~FontStyles.Underline;
}
return true;
case 43045: // <mark=#FF00FF80>
case 30245: // <MARK>
m_style |= FontStyles.Highlight;
m_fontStyleStack.Add(FontStyles.Highlight);
m_highlightColor = HexCharsToColor(m_htmlTag, m_xmlAttribute[0].valueStartIndex, m_xmlAttribute[0].valueLength);
m_highlightColor.a = m_htmlColor.a < m_highlightColor.a ? (byte)(m_htmlColor.a) : (byte)(m_highlightColor.a);
m_highlightColorStack.Add(m_highlightColor);
return true;
case 155892: // </mark>
case 143092: // </MARK>
if ((m_fontStyle & FontStyles.Highlight) != FontStyles.Highlight)
{
m_highlightColor = m_highlightColorStack.Remove();
if (m_fontStyleStack.Remove(FontStyles.Highlight) == 0)
m_style &= ~FontStyles.Highlight;
}
return true;
case 6552: // <sub>
case 4728: // <SUB>
m_fontScaleMultiplier *= m_currentFontAsset.fontInfo.SubSize > 0 ? m_currentFontAsset.fontInfo.SubSize : 1;
m_baselineOffsetStack.Push(m_baselineOffset);
m_baselineOffset += m_currentFontAsset.fontInfo.SubscriptOffset * m_fontScale * m_fontScaleMultiplier;
m_fontStyleStack.Add(FontStyles.Subscript);
m_style |= FontStyles.Subscript;
return true;
case 22673: // </sub>
case 20849: // </SUB>
if ((m_style & FontStyles.Subscript) == FontStyles.Subscript)
{
if (m_fontScaleMultiplier < 1)
{
//m_baselineOffset -= m_currentFontAsset.fontInfo.SubscriptOffset * m_fontScale * m_fontScaleMultiplier;
m_baselineOffset = m_baselineOffsetStack.Pop();
m_fontScaleMultiplier /= m_currentFontAsset.fontInfo.SubSize > 0 ? m_currentFontAsset.fontInfo.SubSize : 1;
}
if (m_fontStyleStack.Remove(FontStyles.Subscript) == 0)
m_style &= ~FontStyles.Subscript;
}
return true;
case 6566: // <sup>
case 4742: // <SUP>
m_fontScaleMultiplier *= m_currentFontAsset.fontInfo.SubSize > 0 ? m_currentFontAsset.fontInfo.SubSize : 1;
m_baselineOffsetStack.Push(m_baselineOffset);
m_baselineOffset += m_currentFontAsset.fontInfo.SuperscriptOffset * m_fontScale * m_fontScaleMultiplier;
m_fontStyleStack.Add(FontStyles.Superscript);
m_style |= FontStyles.Superscript;
return true;
case 22687: // </sup>
case 20863: // </SUP>
if ((m_style & FontStyles.Superscript) == FontStyles.Superscript)
{
if (m_fontScaleMultiplier < 1)
{
//m_baselineOffset -= m_currentFontAsset.fontInfo.SuperscriptOffset * m_fontScale * m_fontScaleMultiplier;
m_baselineOffset = m_baselineOffsetStack.Pop();
m_fontScaleMultiplier /= m_currentFontAsset.fontInfo.SubSize > 0 ? m_currentFontAsset.fontInfo.SubSize : 1;
}
if (m_fontStyleStack.Remove(FontStyles.Superscript) == 0)
m_style &= ~FontStyles.Superscript;
}
return true;
case -330774850: // <font-weight>
case 2012149182: // <FONT-WEIGHT>
value = ConvertToFloat(m_htmlTag, m_xmlAttribute[0].valueStartIndex, m_xmlAttribute[0].valueLength);
if (value == -9999) return false;
if ((m_fontStyle & FontStyles.Bold) == FontStyles.Bold)
{
// Nothing happens since Bold is forced on the text.
//m_fontWeight = 700;
return true;
}
// Remove bold style
m_style &= ~FontStyles.Bold;
switch ((int)value)
{
case 100:
m_fontWeightInternal = 100;
break;
case 200:
m_fontWeightInternal = 200;
break;
case 300:
m_fontWeightInternal = 300;
break;
case 400:
m_fontWeightInternal = 400;
break;
case 500:
m_fontWeightInternal = 500;
break;
case 600:
m_fontWeightInternal = 600;
break;
case 700:
m_fontWeightInternal = 700;
m_style |= FontStyles.Bold;
break;
case 800:
m_fontWeightInternal = 800;
break;
case 900:
m_fontWeightInternal = 900;
break;
}
m_fontWeightStack.Add(m_fontWeightInternal);
return true;
case -1885698441: // </font-weight>
case 457225591: // </FONT-WEIGHT>
m_fontWeightInternal = m_fontWeightStack.Remove();
if (m_fontWeightInternal == 400) m_style &= ~FontStyles.Bold;
return true;
case 6380: // <pos=000.00px> <pos=0em> <pos=50%>
case 4556: // <POS>
value = ConvertToFloat(m_htmlTag, m_xmlAttribute[0].valueStartIndex, m_xmlAttribute[0].valueLength);
if (value == -9999) return false;
switch (tagUnits)
{
case TagUnits.Pixels:
//if (m_xmlAttribute[1].nameHashCode == 275917) //
//{
// // left = 3774683
// // right= 136703040
// if (m_xmlAttribute [1].valueHashCode == 136703040)
// {
// // track the endindex so we can return to this character.
// Debug.Log ("align=right startIndex" + endIndex);
// }
//}
m_xAdvance = value;
//m_isIgnoringAlignment = true;
return true;
case TagUnits.FontUnits:
m_xAdvance = value * m_fontScale * m_fontAsset.fontInfo.TabWidth / m_fontAsset.tabSize;
//m_isIgnoringAlignment = true;
return true;
case TagUnits.Percentage:
m_xAdvance = m_marginWidth * value / 100;
//if (m_xmlAttribute[1].nameHashCode == 275917) //
//{
// //Debug.Log ("align tag.");
// // left = 3774683
// // right= 136703040
// if (m_xmlAttribute [1].valueHashCode == 136703040)
// {
// // track the endindex so we can return to this character.
// SaveWordWrappingState (ref m_SavedAlignment, endIndex, m_characterCount);
// }
//}
//m_isIgnoringAlignment = true;
return true;
}
return false;
case 22501: // </pos>
case 20677: // </POS>
/*
// Should retrun the index of where the tag started and adjust xAdvance.
if (m_isParsingText)
{
if (m_SavedAlignment.previous_WordBreak != 0)
{
float current_xAdvance = m_xAdvance;
endIndex = RestoreWordWrappingState (ref m_SavedAlignment);
m_characterCount -= 1;
m_xAdvance -= current_xAdvance - m_xAdvance;
}
m_SavedAlignment.previous_WordBreak = 0;
m_isIgnoringAlignment = false;
}
*/
m_isIgnoringAlignment = false;
return true;
case 16034505: // <voffset>
case 11642281: // <VOFFSET>
value = ConvertToFloat(m_htmlTag, m_xmlAttribute[0].valueStartIndex, m_xmlAttribute[0].valueLength);
if (value == -9999) return false;
switch (tagUnits)
{
case TagUnits.Pixels:
m_baselineOffset = value;
return true;
case TagUnits.FontUnits:
m_baselineOffset = value * m_fontScale * m_fontAsset.fontInfo.Ascender;
return true;
case TagUnits.Percentage:
//m_baselineOffset = m_marginHeight * val / 100;
return false;
}
return false;
case 54741026: // </voffset>
case 50348802: // </VOFFSET>
m_baselineOffset = 0;
return true;
case 43991: // <page>
case 31191: // <PAGE>
// This tag only works when Overflow - Page mode is used.
if (m_overflowMode == TextOverflowModes.Page)
{
m_xAdvance = 0 + tag_LineIndent + tag_Indent;
m_lineOffset = 0;
m_pageNumber += 1;
m_isNewPage = true;
}
return true;
// <BR> tag is now handled inline where it is replaced by a linefeed or \n.
//case 544: // <BR>
//case 800: // <br>
// m_forceLineBreak = true;
// return true;
case 43969: // <nobr>
case 31169: // <NOBR>
m_isNonBreakingSpace = true;
return true;
case 156816: // </nobr>
case 144016: // </NOBR>
m_isNonBreakingSpace = false;
return true;
case 45545: // <size=>
case 32745: // <SIZE>
value = ConvertToFloat(m_htmlTag, m_xmlAttribute[0].valueStartIndex, m_xmlAttribute[0].valueLength);
if (value == -9999) return false;
switch (tagUnits)
{
case TagUnits.Pixels:
if (m_htmlTag[5] == 43) // <size=+00>
{
m_currentFontSize = m_fontSize + value;
m_sizeStack.Add(m_currentFontSize);
m_fontScale = (m_currentFontSize / m_currentFontAsset.fontInfo.PointSize * m_currentFontAsset.fontInfo.Scale * (m_isOrthographic ? 1 : 0.1f));
return true;
}
else if (m_htmlTag[5] == 45) // <size=-00>
{
m_currentFontSize = m_fontSize + value;
m_sizeStack.Add(m_currentFontSize);
m_fontScale = (m_currentFontSize / m_currentFontAsset.fontInfo.PointSize * m_currentFontAsset.fontInfo.Scale * (m_isOrthographic ? 1 : 0.1f));
return true;
}
else // <size=00.0>
{
m_currentFontSize = value;
m_sizeStack.Add(m_currentFontSize);
m_fontScale = (m_currentFontSize / m_currentFontAsset.fontInfo.PointSize * m_currentFontAsset.fontInfo.Scale * (m_isOrthographic ? 1 : 0.1f));
return true;
}
case TagUnits.FontUnits:
m_currentFontSize = m_fontSize * value;
m_sizeStack.Add(m_currentFontSize);
m_fontScale = (m_currentFontSize / m_currentFontAsset.fontInfo.PointSize * m_currentFontAsset.fontInfo.Scale * (m_isOrthographic ? 1 : 0.1f));
return true;
case TagUnits.Percentage:
m_currentFontSize = m_fontSize * value / 100;
m_sizeStack.Add(m_currentFontSize);
m_fontScale = (m_currentFontSize / m_currentFontAsset.fontInfo.PointSize * m_currentFontAsset.fontInfo.Scale * (m_isOrthographic ? 1 : 0.1f));
return true;
}
return false;
case 158392: // </size>
case 145592: // </SIZE>
m_currentFontSize = m_sizeStack.Remove();
m_fontScale = (m_currentFontSize / m_currentFontAsset.fontInfo.PointSize * m_currentFontAsset.fontInfo.Scale * (m_isOrthographic ? 1 : 0.1f));
return true;
case 41311: // <font=xx>
case 28511: // <FONT>
//Debug.Log("Font name: \"" + new string(m_htmlTag, m_xmlAttribute[0].valueStartIndex, m_xmlAttribute[0].valueLength) + "\" HashCode: " + m_xmlAttribute[0].valueHashCode + " Material Name: \"" + new string(m_htmlTag, m_xmlAttribute[1].valueStartIndex, m_xmlAttribute[1].valueLength) + "\" Hashcode: " + m_xmlAttribute[1].valueHashCode);
int fontHashCode = m_xmlAttribute[0].valueHashCode;
int materialAttributeHashCode = m_xmlAttribute[1].nameHashCode;
int materialHashCode = m_xmlAttribute[1].valueHashCode;
// Special handling for <font=default> or <font=Default>
if (fontHashCode == 764638571 || fontHashCode == 523367755)
{
m_currentFontAsset = m_materialReferences[0].fontAsset;
m_currentMaterial = m_materialReferences[0].material;
m_currentMaterialIndex = 0;
//Debug.Log("<font=Default> assigning Font Asset [" + m_currentFontAsset.name + "] with Material [" + m_currentMaterial.name + "].");
m_fontScale = (m_currentFontSize / m_currentFontAsset.fontInfo.PointSize * m_currentFontAsset.fontInfo.Scale * (m_isOrthographic ? 1 : 0.1f));
m_materialReferenceStack.Add(m_materialReferences[0]);
return true;
}
TMP_FontAsset tempFont;
Material tempMaterial;
// HANDLE NEW FONT ASSET
if (MaterialReferenceManager.TryGetFontAsset(fontHashCode, out tempFont))
{
//if (tempFont != m_currentFontAsset)
//{
// //Debug.Log("Assigning Font Asset: " + tempFont.name);
// m_currentFontAsset = tempFont;
// m_fontScale = (m_currentFontSize / m_currentFontAsset.fontInfo.PointSize * m_currentFontAsset.fontInfo.Scale * (m_isOrthographic ? 1 : 0.1f));
//}
}
else
{
// Load Font Asset
tempFont = Resources.Load<TMP_FontAsset>(TMP_Settings.defaultFontAssetPath + new string(m_htmlTag, m_xmlAttribute[0].valueStartIndex, m_xmlAttribute[0].valueLength));
if (tempFont == null)
return false;
// Add new reference to the font asset as well as default material to the MaterialReferenceManager
MaterialReferenceManager.AddFontAsset(tempFont);
}
// HANDLE NEW MATERIAL
if (materialAttributeHashCode == 0 && materialHashCode == 0)
{
// No material specified then use default font asset material.
m_currentMaterial = tempFont.material;
m_currentMaterialIndex = MaterialReference.AddMaterialReference(m_currentMaterial, tempFont, m_materialReferences, m_materialReferenceIndexLookup);
m_materialReferenceStack.Add(m_materialReferences[m_currentMaterialIndex]);
}
else if (materialAttributeHashCode == 103415287 || materialAttributeHashCode == 72669687) // using material attribute
{
if (MaterialReferenceManager.TryGetMaterial(materialHashCode, out tempMaterial))
{
m_currentMaterial = tempMaterial;
m_currentMaterialIndex = MaterialReference.AddMaterialReference(m_currentMaterial, tempFont, m_materialReferences, m_materialReferenceIndexLookup);
m_materialReferenceStack.Add(m_materialReferences[m_currentMaterialIndex]);
}
else
{
// Load new material
tempMaterial = Resources.Load<Material>(TMP_Settings.defaultFontAssetPath + new string(m_htmlTag, m_xmlAttribute[1].valueStartIndex, m_xmlAttribute[1].valueLength));
if (tempMaterial == null)
return false;
// Add new reference to this material in the MaterialReferenceManager
MaterialReferenceManager.AddFontMaterial(materialHashCode, tempMaterial);
m_currentMaterial = tempMaterial;
m_currentMaterialIndex = MaterialReference.AddMaterialReference(m_currentMaterial, tempFont, m_materialReferences, m_materialReferenceIndexLookup);
m_materialReferenceStack.Add(m_materialReferences[m_currentMaterialIndex]);
}
}
else
return false;
m_currentFontAsset = tempFont;
m_fontScale = (m_currentFontSize / m_currentFontAsset.fontInfo.PointSize * m_currentFontAsset.fontInfo.Scale * (m_isOrthographic ? 1 : 0.1f));
return true;
case 154158: // </font>
case 141358: // </FONT>
{
MaterialReference materialReference = m_materialReferenceStack.Remove();
m_currentFontAsset = materialReference.fontAsset;
m_currentMaterial = materialReference.material;
m_currentMaterialIndex = materialReference.index;
m_fontScale = (m_currentFontSize / m_currentFontAsset.fontInfo.PointSize * m_currentFontAsset.fontInfo.Scale * (m_isOrthographic ? 1 : 0.1f));
return true;
}
case 103415287: // <material="material name">
case 72669687: // <MATERIAL>
materialHashCode = m_xmlAttribute[0].valueHashCode;
// Special handling for <material=default> or <material=Default>
if (materialHashCode == 764638571 || materialHashCode == 523367755)
{
// Check if material font atlas texture matches that of the current font asset.
//if (m_currentFontAsset.atlas.GetInstanceID() != m_currentMaterial.GetTexture(ShaderUtilities.ID_MainTex).GetInstanceID()) return false;
m_currentMaterial = m_materialReferences[0].material;
m_currentMaterialIndex = 0;
m_materialReferenceStack.Add(m_materialReferences[0]);
return true;
}
// Check if material
if (MaterialReferenceManager.TryGetMaterial(materialHashCode, out tempMaterial))
{
// Check if material font atlas texture matches that of the current font asset.
//if (m_currentFontAsset.atlas.GetInstanceID() != tempMaterial.GetTexture(ShaderUtilities.ID_MainTex).GetInstanceID()) return false;
m_currentMaterial = tempMaterial;
m_currentMaterialIndex = MaterialReference.AddMaterialReference(m_currentMaterial, m_currentFontAsset, m_materialReferences, m_materialReferenceIndexLookup);
m_materialReferenceStack.Add(m_materialReferences[m_currentMaterialIndex]);
}
else
{
// Load new material
tempMaterial = Resources.Load<Material>(TMP_Settings.defaultFontAssetPath + new string(m_htmlTag, m_xmlAttribute[0].valueStartIndex, m_xmlAttribute[0].valueLength));
if (tempMaterial == null)
return false;
// Check if material font atlas texture matches that of the current font asset.
//if (m_currentFontAsset.atlas.GetInstanceID() != tempMaterial.GetTexture(ShaderUtilities.ID_MainTex).GetInstanceID()) return false;
// Add new reference to this material in the MaterialReferenceManager
MaterialReferenceManager.AddFontMaterial(materialHashCode, tempMaterial);
m_currentMaterial = tempMaterial;
m_currentMaterialIndex = MaterialReference.AddMaterialReference(m_currentMaterial, m_currentFontAsset , m_materialReferences, m_materialReferenceIndexLookup);
m_materialReferenceStack.Add(m_materialReferences[m_currentMaterialIndex]);
}
return true;
case 374360934: // </material>
case 343615334: // </MATERIAL>
{
//if (m_currentMaterial.GetTexture(ShaderUtilities.ID_MainTex).GetInstanceID() != m_materialReferenceStack.PreviousItem().material.GetTexture(ShaderUtilities.ID_MainTex).GetInstanceID())
// return false;
MaterialReference materialReference = m_materialReferenceStack.Remove();
m_currentMaterial = materialReference.material;
m_currentMaterialIndex = materialReference.index;
return true;
}
case 320078: // <space=000.00>
case 230446: // <SPACE>
value = ConvertToFloat(m_htmlTag, m_xmlAttribute[0].valueStartIndex, m_xmlAttribute[0].valueLength);
if (value == -9999) return false;
switch (tagUnits)
{
case TagUnits.Pixels:
m_xAdvance += value;
return true;
case TagUnits.FontUnits:
m_xAdvance += value * m_fontScale * m_fontAsset.fontInfo.TabWidth / m_fontAsset.tabSize;
return true;
case TagUnits.Percentage:
// Not applicable
return false;
}
return false;
case 276254: // <alpha=#FF>
case 186622: // <ALPHA>
if (m_xmlAttribute[0].valueLength != 3) return false;
m_htmlColor.a = (byte)(HexToInt(m_htmlTag[7]) * 16 + HexToInt(m_htmlTag[8]));
return true;
case 1750458: // <a name=" ">
return false;
case 426: // </a>
return true;
case 43066: // <link="name">
case 30266: // <LINK>
if (m_isParsingText && !m_isCalculatingPreferredValues)
{
int index = m_textInfo.linkCount;
if (index + 1 > m_textInfo.linkInfo.Length)
TMP_TextInfo.Resize(ref m_textInfo.linkInfo, index + 1);
m_textInfo.linkInfo[index].textComponent = this;
m_textInfo.linkInfo[index].hashCode = m_xmlAttribute[0].valueHashCode;
m_textInfo.linkInfo[index].linkTextfirstCharacterIndex = m_characterCount;
m_textInfo.linkInfo[index].linkIdFirstCharacterIndex = startIndex + m_xmlAttribute[0].valueStartIndex;
m_textInfo.linkInfo[index].linkIdLength = m_xmlAttribute[0].valueLength;
m_textInfo.linkInfo[index].SetLinkID(m_htmlTag, m_xmlAttribute[0].valueStartIndex, m_xmlAttribute[0].valueLength);
}
return true;
case 155913: // </link>
case 143113: // </LINK>
if (m_isParsingText && !m_isCalculatingPreferredValues)
{
if (m_textInfo.linkCount < m_textInfo.linkInfo.Length)
{
m_textInfo.linkInfo[m_textInfo.linkCount].linkTextLength = m_characterCount - m_textInfo.linkInfo[m_textInfo.linkCount].linkTextfirstCharacterIndex;
m_textInfo.linkCount += 1;
}
}
return true;
case 275917: // <align=>
case 186285: // <ALIGN>
switch (m_xmlAttribute[0].valueHashCode)
{
case 3774683: // <align=left>
m_lineJustification = TextAlignmentOptions.Left;
m_lineJustificationStack.Add(m_lineJustification);
return true;
case 136703040: // <align=right>
m_lineJustification = TextAlignmentOptions.Right;
m_lineJustificationStack.Add(m_lineJustification);
return true;
case -458210101: // <align=center>
m_lineJustification = TextAlignmentOptions.Center;
m_lineJustificationStack.Add(m_lineJustification);
return true;
case -523808257: // <align=justified>
m_lineJustification = TextAlignmentOptions.Justified;
m_lineJustificationStack.Add(m_lineJustification);
return true;
case 122383428: // <align=flush>
m_lineJustification = TextAlignmentOptions.Flush;
m_lineJustificationStack.Add(m_lineJustification);
return true;
}
return false;
case 1065846: // </align>
case 976214: // </ALIGN>
m_lineJustification = m_lineJustificationStack.Remove();
return true;
case 327550: // <width=xx>
case 237918: // <WIDTH>
value = ConvertToFloat(m_htmlTag, m_xmlAttribute[0].valueStartIndex, m_xmlAttribute[0].valueLength);
if (value == -9999) return false;
switch (tagUnits)
{
case TagUnits.Pixels:
m_width = value;
break;
case TagUnits.FontUnits:
return false;
//break;
case TagUnits.Percentage:
m_width = m_marginWidth * value / 100;
break;
}
return true;
case 1117479: // </width>
case 1027847: // </WIDTH>
m_width = -1;
return true;
// STYLE tag is now handled inline and replaced by its definition.
//case 322689: // <style="name">
//case 233057: // <STYLE>
// TMP_Style style = TMP_StyleSheet.GetStyle(m_xmlAttribute[0].valueHashCode);
// if (style == null) return false;
// m_styleStack.Add(style.hashCode);
// // Parse Style Macro
// for (int i = 0; i < style.styleOpeningTagArray.Length; i++)
// {
// if (style.styleOpeningTagArray[i] == 60)
// {
// if (ValidateHtmlTag(style.styleOpeningTagArray, i + 1, out i) == false) return false;
// }
// }
// return true;
//case 1112618: // </style>
//case 1022986: // </STYLE>
// style = TMP_StyleSheet.GetStyle(m_xmlAttribute[0].valueHashCode);
// if (style == null)
// {
// // Get style from the Style Stack
// int styleHashCode = m_styleStack.CurrentItem();
// style = TMP_StyleSheet.GetStyle(styleHashCode);
// m_styleStack.Remove();
// }
// if (style == null) return false;
// //// Parse Style Macro
// for (int i = 0; i < style.styleClosingTagArray.Length; i++)
// {
// if (style.styleClosingTagArray[i] == 60)
// ValidateHtmlTag(style.styleClosingTagArray, i + 1, out i);
// }
// return true;
case 281955: // <color> <color=#FF00FF> or <color=#FF00FF00>
case 192323: // <COLOR=#FF00FF>
// <color=#FFF> 3 Hex (short hand)
if (m_htmlTag[6] == 35 && tagCharCount == 10)
{
m_htmlColor = HexCharsToColor(m_htmlTag, tagCharCount);
m_colorStack.Add(m_htmlColor);
return true;
}
// <color=#FFF7> 4 Hex (short hand)
else if (m_htmlTag[6] == 35 && tagCharCount == 11)
{
m_htmlColor = HexCharsToColor(m_htmlTag, tagCharCount);
m_colorStack.Add(m_htmlColor);
return true;
}
// <color=#FF00FF> 3 Hex pairs
if (m_htmlTag[6] == 35 && tagCharCount == 13)
{
m_htmlColor = HexCharsToColor(m_htmlTag, tagCharCount);
m_colorStack.Add(m_htmlColor);
return true;
}
// <color=#FF00FF00> 4 Hex pairs
else if (m_htmlTag[6] == 35 && tagCharCount == 15)
{
m_htmlColor = HexCharsToColor(m_htmlTag, tagCharCount);
m_colorStack.Add(m_htmlColor);
return true;
}
// <color=name>
switch (m_xmlAttribute[0].valueHashCode)
{
case 125395: // <color=red>
m_htmlColor = Color.red;
m_colorStack.Add(m_htmlColor);
return true;
case 3573310: // <color=blue>
m_htmlColor = Color.blue;
m_colorStack.Add(m_htmlColor);
return true;
case 117905991: // <color=black>
m_htmlColor = Color.black;
m_colorStack.Add(m_htmlColor);
return true;
case 121463835: // <color=green>
m_htmlColor = Color.green;
m_colorStack.Add(m_htmlColor);
return true;
case 140357351: // <color=white>
m_htmlColor = Color.white;
m_colorStack.Add(m_htmlColor);
return true;
case 26556144: // <color=orange>
m_htmlColor = new Color32(255, 128, 0, 255);
m_colorStack.Add(m_htmlColor);
return true;
case -36881330: // <color=purple>
m_htmlColor = new Color32(160, 32, 240, 255);
m_colorStack.Add(m_htmlColor);
return true;
case 554054276: // <color=yellow>
m_htmlColor = Color.yellow;
m_colorStack.Add(m_htmlColor);
return true;
}
return false;
case 100149144: //<gradient>
case 69403544: // <GRADIENT>
int gradientPresetHashCode = m_xmlAttribute[0].valueHashCode;
TMP_ColorGradient tempColorGradientPreset;
// Check if Color Gradient Preset has already been loaded.
if (MaterialReferenceManager.TryGetColorGradientPreset(gradientPresetHashCode, out tempColorGradientPreset))
{
m_colorGradientPreset = tempColorGradientPreset;
}
else
{
// Load Color Gradient Preset
if (tempColorGradientPreset == null)
{
tempColorGradientPreset = Resources.Load<TMP_ColorGradient>(TMP_Settings.defaultColorGradientPresetsPath + new string(m_htmlTag, m_xmlAttribute[0].valueStartIndex, m_xmlAttribute[0].valueLength));
}
if (tempColorGradientPreset == null)
return false;
MaterialReferenceManager.AddColorGradientPreset(gradientPresetHashCode, tempColorGradientPreset);
m_colorGradientPreset = tempColorGradientPreset;
}
m_colorGradientStack.Add(m_colorGradientPreset);
// TODO : Add support for defining preset in the tag itself
return true;
case 371094791: // </gradient>
case 340349191: // </GRADIENT>
m_colorGradientPreset = m_colorGradientStack.Remove();
return true;
case 1983971: // <cspace=xx.x>
case 1356515: // <CSPACE>
value = ConvertToFloat(m_htmlTag, m_xmlAttribute[0].valueStartIndex, m_xmlAttribute[0].valueLength);
if (value == -9999) return false;
switch (tagUnits)
{
case TagUnits.Pixels:
m_cSpacing = value;
break;
case TagUnits.FontUnits:
m_cSpacing = value;
m_cSpacing *= m_fontScale * m_fontAsset.fontInfo.TabWidth / m_fontAsset.tabSize;
break;
case TagUnits.Percentage:
return false;
}
return true;
case 7513474: // </cspace>
case 6886018: // </CSPACE>
if (!m_isParsingText) return true;
// Adjust xAdvance to remove extra space from last character.
if (m_characterCount > 0)
{
m_xAdvance -= m_cSpacing;
m_textInfo.characterInfo[m_characterCount - 1].xAdvance = m_xAdvance;
}
m_cSpacing = 0;
return true;
case 2152041: // <mspace=xx.x>
case 1524585: // <MSPACE>
value = ConvertToFloat(m_htmlTag, m_xmlAttribute[0].valueStartIndex, m_xmlAttribute[0].valueLength);
if (value == -9999) return false;
switch (tagUnits)
{
case TagUnits.Pixels:
m_monoSpacing = value;
break;
case TagUnits.FontUnits:
m_monoSpacing = value;
m_monoSpacing *= m_fontScale * m_fontAsset.fontInfo.TabWidth / m_fontAsset.tabSize;
break;
case TagUnits.Percentage:
return false;
}
return true;
case 7681544: // </mspace>
case 7054088: // </MSPACE>
m_monoSpacing = 0;
return true;
case 280416: // <class="name">
return false;
case 1071884: // </color>
case 982252: // </COLOR>
m_htmlColor = m_colorStack.Remove();
return true;
case 2068980: // <indent=10px> <indent=10em> <indent=50%>
case 1441524: // <INDENT>
value = ConvertToFloat(m_htmlTag, m_xmlAttribute[0].valueStartIndex, m_xmlAttribute[0].valueLength);
if (value == -9999) return false;
switch (tagUnits)
{
case TagUnits.Pixels:
tag_Indent = value;
break;
case TagUnits.FontUnits:
tag_Indent = value;
tag_Indent *= m_fontScale * m_fontAsset.fontInfo.TabWidth / m_fontAsset.tabSize;
break;
case TagUnits.Percentage:
tag_Indent = m_marginWidth * value / 100;
break;
}
m_indentStack.Add(tag_Indent);
m_xAdvance = tag_Indent;
return true;
case 7598483: // </indent>
case 6971027: // </INDENT>
tag_Indent = m_indentStack.Remove();
//m_xAdvance = tag_Indent;
return true;
case 1109386397: // <line-indent>
case -842656867: // <LINE-INDENT>
value = ConvertToFloat(m_htmlTag, m_xmlAttribute[0].valueStartIndex, m_xmlAttribute[0].valueLength);
if (value == -9999) return false;
switch (tagUnits)
{
case TagUnits.Pixels:
tag_LineIndent = value;
break;
case TagUnits.FontUnits:
tag_LineIndent = value;
tag_LineIndent *= m_fontScale * m_fontAsset.fontInfo.TabWidth / m_fontAsset.tabSize;
break;
case TagUnits.Percentage:
tag_LineIndent = m_marginWidth * value / 100;
break;
}
m_xAdvance += tag_LineIndent;
return true;
case -445537194: // </line-indent>
case 1897386838: // </LINE-INDENT>
tag_LineIndent = 0;
return true;
case 2246877: // <sprite=x>
case 1619421: // <SPRITE>
int spriteAssetHashCode = m_xmlAttribute[0].valueHashCode;
TMP_SpriteAsset tempSpriteAsset;
m_spriteIndex = -1;
// CHECK TAG FORMAT
if (m_xmlAttribute[0].valueType == TagType.None || m_xmlAttribute[0].valueType == TagType.NumericalValue)
{
// No Sprite Asset is assigned to the text object
if (m_spriteAsset != null)
{
m_currentSpriteAsset = m_spriteAsset;
}
else if (m_defaultSpriteAsset != null)
{
m_currentSpriteAsset = m_defaultSpriteAsset;
}
else if (m_defaultSpriteAsset == null)
{
if (TMP_Settings.defaultSpriteAsset != null)
m_defaultSpriteAsset = TMP_Settings.defaultSpriteAsset;
else
m_defaultSpriteAsset = Resources.Load<TMP_SpriteAsset>("Sprite Assets/Default Sprite Asset");
m_currentSpriteAsset = m_defaultSpriteAsset;
}
// No valid sprite asset available
if (m_currentSpriteAsset == null)
return false;
}
else
{
// A Sprite Asset has been specified
if (MaterialReferenceManager.TryGetSpriteAsset(spriteAssetHashCode, out tempSpriteAsset))
{
m_currentSpriteAsset = tempSpriteAsset;
}
else
{
// Load Sprite Asset
if (tempSpriteAsset == null)
{
tempSpriteAsset = Resources.Load<TMP_SpriteAsset>(TMP_Settings.defaultSpriteAssetPath + new string(m_htmlTag, m_xmlAttribute[0].valueStartIndex, m_xmlAttribute[0].valueLength));
}
if (tempSpriteAsset == null)
return false;
//Debug.Log("Loading & assigning new Sprite Asset: " + tempSpriteAsset.name);
MaterialReferenceManager.AddSpriteAsset(spriteAssetHashCode, tempSpriteAsset);
m_currentSpriteAsset = tempSpriteAsset;
}
}
// Handling of <sprite=index> legacy tag format.
if (m_xmlAttribute[0].valueType == TagType.NumericalValue) // <sprite=index>
{
int index = (int)ConvertToFloat(m_htmlTag, m_xmlAttribute[0].valueStartIndex, m_xmlAttribute[0].valueLength);
if (index == -9999) return false;
// Check to make sure sprite index is valid
if (index > m_currentSpriteAsset.spriteInfoList.Count - 1) return false;
m_spriteIndex = index;
}
m_spriteColor = s_colorWhite;
m_tintSprite = false;
// Handle Sprite Tag Attributes
for (int i = 0; i < m_xmlAttribute.Length && m_xmlAttribute[i].nameHashCode != 0; i++)
{
//Debug.Log("Attribute[" + i + "].nameHashCode=" + m_xmlAttribute[i].nameHashCode + " Value:" + ConvertToFloat(m_htmlTag, m_xmlAttribute[i].valueStartIndex, m_xmlAttribute[i].valueLength));
int nameHashCode = m_xmlAttribute[i].nameHashCode;
int index = 0;
switch (nameHashCode)
{
case 43347: // <sprite name="">
case 30547: // <SPRITE NAME="">
m_currentSpriteAsset = TMP_SpriteAsset.SearchForSpriteByHashCode(m_currentSpriteAsset, m_xmlAttribute[i].valueHashCode, true, out index);
if (index == -1) return false;
m_spriteIndex = index;
break;
case 295562: // <sprite index=>
case 205930: // <SPRITE INDEX=>
index = (int)ConvertToFloat(m_htmlTag, m_xmlAttribute[1].valueStartIndex, m_xmlAttribute[1].valueLength);
if (index == -9999) return false;
// Check to make sure sprite index is valid
if (index > m_currentSpriteAsset.spriteInfoList.Count - 1) return false;
m_spriteIndex = index;
break;
case 45819: // tint
case 33019: // TINT
m_tintSprite = ConvertToFloat(m_htmlTag, m_xmlAttribute[i].valueStartIndex, m_xmlAttribute[i].valueLength) != 0;
break;
case 281955: // color=#FF00FF80
case 192323: // COLOR
m_spriteColor = HexCharsToColor(m_htmlTag, m_xmlAttribute[i].valueStartIndex, m_xmlAttribute[i].valueLength);
break;
case 39505: // anim="0,16,12" start, end, fps
case 26705: // ANIM
//Debug.Log("Start: " + m_xmlAttribute[i].valueStartIndex + " Length: " + m_xmlAttribute[i].valueLength);
int paramCount = GetAttributeParameters(m_htmlTag, m_xmlAttribute[i].valueStartIndex, m_xmlAttribute[i].valueLength, ref m_attributeParameterValues);
if (paramCount != 3) return false;
m_spriteIndex = (int)m_attributeParameterValues[0];
if (m_isParsingText)
{
// TODO : fix this!
//if (m_attributeParameterValues[0] > m_currentSpriteAsset.spriteInfoList.Count - 1 || m_attributeParameterValues[1] > m_currentSpriteAsset.spriteInfoList.Count - 1)
// return false;
spriteAnimator.DoSpriteAnimation(m_characterCount, m_currentSpriteAsset, m_spriteIndex, (int)m_attributeParameterValues[1], (int)m_attributeParameterValues[2]);
}
break;
//case 45545: // size
//case 32745: // SIZE
// break;
default:
if (nameHashCode != 2246877 && nameHashCode != 1619421)
return false;
break;
}
}
if (m_spriteIndex == -1) return false;
// Material HashCode for the Sprite Asset is the Sprite Asset Hash Code
m_currentMaterialIndex = MaterialReference.AddMaterialReference(m_currentSpriteAsset.material, m_currentSpriteAsset, m_materialReferences, m_materialReferenceIndexLookup);
m_textElementType = TMP_TextElementType.Sprite;
return true;
case 730022849: // <lowercase>
case 514803617: // <LOWERCASE>
m_style |= FontStyles.LowerCase;
m_fontStyleStack.Add(FontStyles.LowerCase);
return true;
case -1668324918: // </lowercase>
case -1883544150: // </LOWERCASE>
if (m_fontStyleStack.Remove(FontStyles.LowerCase) == 0)
m_style &= ~FontStyles.LowerCase;
return true;
case 13526026: // <allcaps>
case 9133802: // <ALLCAPS>
case 781906058: // <uppercase>
case 566686826: // <UPPERCASE>
m_style |= FontStyles.UpperCase;
m_fontStyleStack.Add(FontStyles.UpperCase);
return true;
case 52232547: // </allcaps>
case 47840323: // </ALLCAPS>
case -1616441709: // </uppercase>
case -1831660941: // </UPPERCASE>
if (m_fontStyleStack.Remove(FontStyles.UpperCase) == 0)
m_style &= ~FontStyles.UpperCase;
return true;
case 766244328: // <smallcaps>
case 551025096: // <SMALLCAPS>
m_style |= FontStyles.SmallCaps;
m_fontStyleStack.Add(FontStyles.SmallCaps);
return true;
case -1632103439: // </smallcaps>
case -1847322671: // </SMALLCAPS>
if (m_fontStyleStack.Remove(FontStyles.SmallCaps) == 0)
m_style &= ~FontStyles.SmallCaps;
return true;
case 2109854: // <margin=00.0> <margin=00em> <margin=50%>
case 1482398: // <MARGIN>
value = ConvertToFloat(m_htmlTag, m_xmlAttribute[0].valueStartIndex, m_xmlAttribute[0].valueLength); // px
if (value == -9999) return false;
m_marginLeft = value;
switch (tagUnits)
{
case TagUnits.Pixels:
// Default behavior
break;
case TagUnits.FontUnits:
m_marginLeft *= m_fontScale * m_fontAsset.fontInfo.TabWidth / m_fontAsset.tabSize;
break;
case TagUnits.Percentage:
m_marginLeft = (m_marginWidth - (m_width != -1 ? m_width : 0)) * m_marginLeft / 100;
break;
}
m_marginLeft = m_marginLeft >= 0 ? m_marginLeft : 0;
m_marginRight = m_marginLeft;
return true;
case 7639357: // </margin>
case 7011901: // </MARGIN>
m_marginLeft = 0;
m_marginRight = 0;
return true;
case 1100728678: // <margin-left=xx.x>
case -855002522: // <MARGIN-LEFT>
value = ConvertToFloat(m_htmlTag, m_xmlAttribute[0].valueStartIndex, m_xmlAttribute[0].valueLength); // px
if (value == -9999) return false;
m_marginLeft = value;
switch (tagUnits)
{
case TagUnits.Pixels:
// Default behavior
break;
case TagUnits.FontUnits:
m_marginLeft *= m_fontScale * m_fontAsset.fontInfo.TabWidth / m_fontAsset.tabSize;
break;
case TagUnits.Percentage:
m_marginLeft = (m_marginWidth - (m_width != -1 ? m_width : 0)) * m_marginLeft / 100;
break;
}
m_marginLeft = m_marginLeft >= 0 ? m_marginLeft : 0;
return true;
case -884817987: // <margin-right=xx.x>
case -1690034531: // <MARGIN-RIGHT>
value = ConvertToFloat(m_htmlTag, m_xmlAttribute[0].valueStartIndex, m_xmlAttribute[0].valueLength); // px
if (value == -9999) return false;
m_marginRight = value;
switch (tagUnits)
{
case TagUnits.Pixels:
// Default behavior
break;
case TagUnits.FontUnits:
m_marginRight *= m_fontScale * m_fontAsset.fontInfo.TabWidth / m_fontAsset.tabSize;
break;
case TagUnits.Percentage:
m_marginRight = (m_marginWidth - (m_width != -1 ? m_width : 0)) * m_marginRight / 100;
break;
}
m_marginRight = m_marginRight >= 0 ? m_marginRight : 0;
return true;
case 1109349752: // <line-height=xx.x>
case -842693512: // <LINE-HEIGHT>
value = ConvertToFloat(m_htmlTag, m_xmlAttribute[0].valueStartIndex, m_xmlAttribute[0].valueLength);
if (value == -9999 || value == 0) return false;
m_lineHeight = value;
switch (tagUnits)
{
case TagUnits.Pixels:
//m_lineHeight = value;
break;
case TagUnits.FontUnits:
m_lineHeight *= m_fontAsset.fontInfo.LineHeight * m_fontScale;
break;
case TagUnits.Percentage:
m_lineHeight = m_fontAsset.fontInfo.LineHeight * m_lineHeight / 100 * m_fontScale;
break;
}
return true;
case -445573839: // </line-height>
case 1897350193: // </LINE-HEIGHT>
m_lineHeight = TMP_Math.FLOAT_UNSET;
return true;
case 15115642: // <noparse>
case 10723418: // <NOPARSE>
tag_NoParsing = true;
return true;
case 1913798: // <action>
case 1286342: // <ACTION>
int actionID = m_xmlAttribute[0].valueHashCode;
if (m_isParsingText)
{
m_actionStack.Add(actionID);
Debug.Log("Action ID: [" + actionID + "] First character index: " + m_characterCount);
}
//if (m_isParsingText)
//{
// TMP_Action action = TMP_Action.GetAction(m_xmlAttribute[0].valueHashCode);
//}
return true;
case 7443301: // </action>
case 6815845: // </ACTION>
if (m_isParsingText)
{
Debug.Log("Action ID: [" + m_actionStack.CurrentItem() + "] Last character index: " + (m_characterCount - 1));
}
m_actionStack.Remove();
return true;
case 315682: // <scale=xx.x>
case 226050: // <SCALE=xx.x>
value = ConvertToFloat(m_htmlTag, m_xmlAttribute[0].valueStartIndex, m_xmlAttribute[0].valueLength);
if (value == -9999) return false;
m_FXMatrix = Matrix4x4.TRS(Vector3.zero, Quaternion.identity, new Vector3(value, 1, 1));
m_isFXMatrixSet = true;
return true;
case 1105611: // </scale>
case 1015979: // </SCALE>
m_isFXMatrixSet = false;
return true;
case 2227963: // <rotate=xx.x>
case 1600507: // <ROTATE=xx.x>
// TODO: Add ability to use Random Rotation
value = ConvertToFloat(m_htmlTag, m_xmlAttribute[0].valueStartIndex, m_xmlAttribute[0].valueLength);
if (value == -9999) return false;
m_FXMatrix = Matrix4x4.TRS(Vector3.zero, Quaternion.Euler(0, 0, value), Vector3.one);
m_isFXMatrixSet = true;
return true;
case 7757466: // </rotate>
case 7130010: // </ROTATE>
m_isFXMatrixSet = false;
return true;
case 317446: // <table>
case 227814: // <TABLE>
switch (m_xmlAttribute[1].nameHashCode)
{
case 327550: // width
float tableWidth = ConvertToFloat(m_htmlTag, m_xmlAttribute[1].valueStartIndex, m_xmlAttribute[1].valueLength);
switch (tagUnits)
{
case TagUnits.Pixels:
Debug.Log("Table width = " + tableWidth + "px.");
break;
case TagUnits.FontUnits:
Debug.Log("Table width = " + tableWidth + "em.");
break;
case TagUnits.Percentage:
Debug.Log("Table width = " + tableWidth + "%.");
break;
}
break;
}
return true;
case 1107375: // </table>
case 1017743: // </TABLE>
return true;
case 926: // <tr>
case 670: // <TR>
return true;
case 3229: // </tr>
case 2973: // </TR>
return true;
case 916: // <th>
case 660: // <TH>
// Set style to bold and center alignment
return true;
case 3219: // </th>
case 2963: // </TH>
return true;
case 912: // <td>
case 656: // <TD>
// Style options
for (int i = 1; i < m_xmlAttribute.Length && m_xmlAttribute[i].nameHashCode != 0; i++)
{
switch (m_xmlAttribute[i].nameHashCode)
{
case 327550: // width
float tableWidth = ConvertToFloat(m_htmlTag, m_xmlAttribute[i].valueStartIndex, m_xmlAttribute[i].valueLength);
switch (tagUnits)
{
case TagUnits.Pixels:
Debug.Log("Table width = " + tableWidth + "px.");
break;
case TagUnits.FontUnits:
Debug.Log("Table width = " + tableWidth + "em.");
break;
case TagUnits.Percentage:
Debug.Log("Table width = " + tableWidth + "%.");
break;
}
break;
case 275917: // align
switch (m_xmlAttribute[i].valueHashCode)
{
case 3774683: // left
Debug.Log("TD align=\"left\".");
break;
case 136703040: // right
Debug.Log("TD align=\"right\".");
break;
case -458210101: // center
Debug.Log("TD align=\"center\".");
break;
case -523808257: // justified
Debug.Log("TD align=\"justified\".");
break;
}
break;
}
}
return true;
case 3215: // </td>
case 2959: // </TD>
return true;
}
}
return false;
}
}
}