TMP_CharacterPropertyDrawer.cs 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235
  1. using UnityEngine;
  2. using UnityEngine.TextCore;
  3. using UnityEngine.TextCore.LowLevel;
  4. using UnityEditor;
  5. using System.Collections;
  6. namespace TMPro.EditorUtilities
  7. {
  8. [CustomPropertyDrawer(typeof(TMP_Character))]
  9. public class TMP_CharacterPropertyDrawer : PropertyDrawer
  10. {
  11. private string k_ColorProperty = "_Color";
  12. int m_GlyphSelectedForEditing = -1;
  13. public override void OnGUI(Rect position, SerializedProperty property, GUIContent label)
  14. {
  15. SerializedProperty prop_Unicode = property.FindPropertyRelative("m_Unicode");
  16. SerializedProperty prop_GlyphIndex = property.FindPropertyRelative("m_GlyphIndex");
  17. SerializedProperty prop_Scale = property.FindPropertyRelative("m_Scale");
  18. GUIStyle style = new GUIStyle(EditorStyles.label);
  19. style.richText = true;
  20. EditorGUIUtility.labelWidth = 40f;
  21. EditorGUIUtility.fieldWidth = 50;
  22. Rect rect = new Rect(position.x + 50, position.y, position.width, 49);
  23. // Display non-editable fields
  24. if (GUI.enabled == false)
  25. {
  26. int unicode = prop_Unicode.intValue;
  27. EditorGUI.LabelField(new Rect(rect.x, rect.y, 120f, 18), new GUIContent("Unicode: <color=#FFFF80>0x" + unicode.ToString("X") + "</color>"), style);
  28. EditorGUI.LabelField(new Rect(rect.x + 115, rect.y, 120f, 18), unicode <= 0xFFFF ? new GUIContent("UTF16: <color=#FFFF80>\\u" + unicode.ToString("X4") + "</color>") : new GUIContent("UTF32: <color=#FFFF80>\\U" + unicode.ToString("X8") + "</color>"), style);
  29. EditorGUI.LabelField(new Rect(rect.x, rect.y + 18, 120, 18), new GUIContent("Glyph ID: <color=#FFFF80>" + prop_GlyphIndex.intValue + "</color>"), style);
  30. EditorGUI.LabelField(new Rect(rect.x, rect.y + 36, 80, 18), new GUIContent("Scale: <color=#FFFF80>" + prop_Scale.floatValue + "</color>"), style);
  31. // Draw Glyph (if exists)
  32. DrawGlyph(position, property);
  33. }
  34. else // Display editable fields
  35. {
  36. EditorGUIUtility.labelWidth = 55f;
  37. GUI.SetNextControlName("Unicode Input");
  38. EditorGUI.BeginChangeCheck();
  39. string unicode = EditorGUI.TextField(new Rect(rect.x, rect.y, 120, 18), "Unicode:", prop_Unicode.intValue.ToString("X"));
  40. if (GUI.GetNameOfFocusedControl() == "Unicode Input")
  41. {
  42. //Filter out unwanted characters.
  43. char chr = Event.current.character;
  44. if ((chr < '0' || chr > '9') && (chr < 'a' || chr > 'f') && (chr < 'A' || chr > 'F'))
  45. {
  46. Event.current.character = '\0';
  47. }
  48. }
  49. if (EditorGUI.EndChangeCheck())
  50. {
  51. // Update Unicode value
  52. prop_Unicode.intValue = TMP_TextUtilities.StringHexToInt(unicode);
  53. }
  54. // Cache current glyph index in case it needs to be restored if the new glyph index is invalid.
  55. int currentGlyphIndex = prop_GlyphIndex.intValue;
  56. EditorGUIUtility.labelWidth = 59f;
  57. EditorGUI.BeginChangeCheck();
  58. EditorGUI.DelayedIntField(new Rect(rect.x, rect.y + 18, 100, 18), prop_GlyphIndex, new GUIContent("Glyph ID:"));
  59. if (EditorGUI.EndChangeCheck())
  60. {
  61. // Get a reference to the font asset
  62. TMP_FontAsset fontAsset = property.serializedObject.targetObject as TMP_FontAsset;
  63. // Make sure new glyph index is valid.
  64. int elementIndex = fontAsset.glyphTable.FindIndex(item => item.index == prop_GlyphIndex.intValue);
  65. if (elementIndex == -1)
  66. prop_GlyphIndex.intValue = currentGlyphIndex;
  67. else
  68. fontAsset.IsFontAssetLookupTablesDirty = true;
  69. }
  70. int glyphIndex = prop_GlyphIndex.intValue;
  71. // Reset glyph selection if new character has been selected.
  72. if (GUI.enabled && m_GlyphSelectedForEditing != glyphIndex)
  73. m_GlyphSelectedForEditing = -1;
  74. // Display button to edit the glyph data.
  75. if (GUI.Button(new Rect(rect.x + 120, rect.y + 18, 75, 18), new GUIContent("Edit Glyph")))
  76. {
  77. if (m_GlyphSelectedForEditing == -1)
  78. m_GlyphSelectedForEditing = glyphIndex;
  79. else
  80. m_GlyphSelectedForEditing = -1;
  81. // Button clicks should not result in potential change.
  82. GUI.changed = false;
  83. }
  84. // Show the glyph property drawer if selected
  85. if (glyphIndex == m_GlyphSelectedForEditing && GUI.enabled)
  86. {
  87. // Get a reference to the font asset
  88. TMP_FontAsset fontAsset = property.serializedObject.targetObject as TMP_FontAsset;
  89. if (fontAsset != null)
  90. {
  91. // Get the index of the glyph in the font asset glyph table.
  92. int elementIndex = fontAsset.glyphTable.FindIndex(item => item.index == glyphIndex);
  93. if (elementIndex != -1)
  94. {
  95. SerializedProperty prop_GlyphTable = property.serializedObject.FindProperty("m_GlyphTable");
  96. SerializedProperty prop_Glyph = prop_GlyphTable.GetArrayElementAtIndex(elementIndex);
  97. SerializedProperty prop_GlyphMetrics = prop_Glyph.FindPropertyRelative("m_Metrics");
  98. SerializedProperty prop_GlyphRect = prop_Glyph.FindPropertyRelative("m_GlyphRect");
  99. Rect newRect = EditorGUILayout.GetControlRect(false, 115);
  100. EditorGUI.DrawRect(new Rect(newRect.x + 52, newRect.y - 20, newRect.width - 52, newRect.height - 5), new Color(0.1f, 0.1f, 0.1f, 0.45f));
  101. EditorGUI.DrawRect(new Rect(newRect.x + 53, newRect.y - 19, newRect.width - 54, newRect.height - 7), new Color(0.3f, 0.3f, 0.3f, 0.8f));
  102. // Display GlyphRect
  103. newRect.x += 55;
  104. newRect.y -= 18;
  105. newRect.width += 5;
  106. EditorGUI.PropertyField(newRect, prop_GlyphRect);
  107. // Display GlyphMetrics
  108. newRect.y += 45;
  109. EditorGUI.PropertyField(newRect, prop_GlyphMetrics);
  110. rect.y += 120;
  111. }
  112. }
  113. }
  114. EditorGUIUtility.labelWidth = 39f;
  115. EditorGUI.PropertyField(new Rect(rect.x, rect.y + 36, 80, 18), prop_Scale, new GUIContent("Scale:"));
  116. // Draw Glyph (if exists)
  117. DrawGlyph(position, property);
  118. }
  119. }
  120. public override float GetPropertyHeight(SerializedProperty property, GUIContent label)
  121. {
  122. return 58;
  123. }
  124. void DrawGlyph(Rect position, SerializedProperty property)
  125. {
  126. // Get a reference to the atlas texture
  127. TMP_FontAsset fontAsset = property.serializedObject.targetObject as TMP_FontAsset;
  128. if (fontAsset == null)
  129. return;
  130. // Get a reference to the Glyph Table
  131. SerializedProperty prop_GlyphTable = property.serializedObject.FindProperty("m_GlyphTable");
  132. int glyphIndex = property.FindPropertyRelative("m_GlyphIndex").intValue;
  133. int elementIndex = fontAsset.glyphTable.FindIndex(item => item.index == glyphIndex);
  134. // Return if we can't find the glyph
  135. if (elementIndex == -1)
  136. return;
  137. SerializedProperty prop_Glyph = prop_GlyphTable.GetArrayElementAtIndex(elementIndex);
  138. // Get reference to atlas texture.
  139. int atlasIndex = prop_Glyph.FindPropertyRelative("m_AtlasIndex").intValue;
  140. Texture2D atlasTexture = fontAsset.atlasTextures.Length > atlasIndex ? fontAsset.atlasTextures[atlasIndex] : null;
  141. if (atlasTexture == null)
  142. return;
  143. Material mat;
  144. if (((GlyphRasterModes)fontAsset.atlasRenderMode & GlyphRasterModes.RASTER_MODE_BITMAP) == GlyphRasterModes.RASTER_MODE_BITMAP)
  145. {
  146. mat = TMP_FontAssetEditor.internalBitmapMaterial;
  147. if (mat == null)
  148. return;
  149. mat.mainTexture = atlasTexture;
  150. mat.SetColor(k_ColorProperty, Color.white);
  151. }
  152. else
  153. {
  154. mat = TMP_FontAssetEditor.internalSDFMaterial;
  155. if (mat == null)
  156. return;
  157. mat.mainTexture = atlasTexture;
  158. mat.SetFloat(ShaderUtilities.ID_GradientScale, fontAsset.atlasPadding + 1);
  159. }
  160. // Draw glyph
  161. Rect glyphDrawPosition = new Rect(position.x, position.y, 48, 58);
  162. SerializedProperty glyphRectProperty = prop_Glyph.FindPropertyRelative("m_GlyphRect");
  163. int padding = fontAsset.atlasPadding;
  164. int glyphOriginX = glyphRectProperty.FindPropertyRelative("m_X").intValue - padding;
  165. int glyphOriginY = glyphRectProperty.FindPropertyRelative("m_Y").intValue - padding;
  166. int glyphWidth = glyphRectProperty.FindPropertyRelative("m_Width").intValue + padding * 2;
  167. int glyphHeight = glyphRectProperty.FindPropertyRelative("m_Height").intValue + padding * 2;
  168. float normalizedHeight = fontAsset.faceInfo.ascentLine - fontAsset.faceInfo.descentLine;
  169. float scale = glyphDrawPosition.width / normalizedHeight;
  170. // Compute the normalized texture coordinates
  171. Rect texCoords = new Rect((float)glyphOriginX / atlasTexture.width, (float)glyphOriginY / atlasTexture.height, (float)glyphWidth / atlasTexture.width, (float)glyphHeight / atlasTexture.height);
  172. if (Event.current.type == EventType.Repaint)
  173. {
  174. glyphDrawPosition.x += (glyphDrawPosition.width - glyphWidth * scale) / 2;
  175. glyphDrawPosition.y += (glyphDrawPosition.height - glyphHeight * scale) / 2;
  176. glyphDrawPosition.width = glyphWidth * scale;
  177. glyphDrawPosition.height = glyphHeight * scale;
  178. // Could switch to using the default material of the font asset which would require passing scale to the shader.
  179. Graphics.DrawTexture(glyphDrawPosition, atlasTexture, texCoords, 0, 0, 0, 0, new Color(1f, 1f, 1f), mat);
  180. }
  181. }
  182. }
  183. }