UILable添加预定义字体样式选项

作者:追风剑情 发布于:2017-8-22 14:34 分类:NGUI

NGUI-3.8

一、修改UILabel.cs

  1. //-------------------------------------------------
  2. // NGUI: Next-Gen UI kit
  3. // Copyright © 2011-2017 Tasharen Entertainment Inc
  4. //-------------------------------------------------
  5.  
  6. using UnityEngine;
  7. using System.Collections.Generic;
  8. using System;
  9. using Alignment = NGUIText.Alignment;
  10. #if UNITY_EDITOR
  11. using System.IO;
  12. using LitJson;
  13. #endif
  14.  
  15. [ExecuteInEditMode]
  16. [AddComponentMenu("NGUI/UI/NGUI Label")]
  17. public class UILabel : UIWidget
  18. {
  19. public enum Effect
  20. {
  21. None,
  22. Shadow,
  23. Outline,
  24. Outline8,
  25. }
  26.  
  27. public enum Overflow
  28. {
  29. ShrinkContent,
  30. ClampContent,
  31. ResizeFreely,
  32. ResizeHeight,
  33. }
  34.  
  35. public enum Crispness
  36. {
  37. Never,
  38. OnDesktop,
  39. Always,
  40. }
  41.  
  42. public enum Modifier
  43. {
  44. None,
  45. ToUppercase,
  46. ToLowercase,
  47. Custom = 255,
  48. }
  49.  
  50. /// <summary>
  51. /// Whether the label will keep its content crisp even when shrunk.
  52. /// You may want to turn this off on mobile devices.
  53. /// </summary>
  54.  
  55. public Crispness keepCrispWhenShrunk = Crispness.OnDesktop;
  56.  
  57. [HideInInspector][SerializeField] Font mTrueTypeFont;
  58. [HideInInspector][SerializeField] UIFont mFont;
  59. #if !UNITY_3_5
  60. [MultilineAttribute(6)]
  61. #endif
  62. [HideInInspector][SerializeField] string mText = "";
  63. [HideInInspector][SerializeField] int mFontSize = 16;
  64. [HideInInspector][SerializeField] FontStyle mFontStyle = FontStyle.Normal;
  65. [HideInInspector][SerializeField] Alignment mAlignment = Alignment.Automatic;
  66. [HideInInspector][SerializeField] bool mEncoding = true;
  67. [HideInInspector][SerializeField] int mMaxLineCount = 0; // 0 denotes unlimited
  68. [HideInInspector][SerializeField] Effect mEffectStyle = Effect.None;
  69. [HideInInspector][SerializeField] Color mEffectColor = Color.black;
  70. [HideInInspector][SerializeField] NGUIText.SymbolStyle mSymbols = NGUIText.SymbolStyle.Normal;
  71. [HideInInspector][SerializeField] Vector2 mEffectDistance = Vector2.one;
  72. [HideInInspector][SerializeField] Overflow mOverflow = Overflow.ShrinkContent;
  73. [HideInInspector][SerializeField] bool mApplyGradient = false;
  74. [HideInInspector][SerializeField] Color mGradientTop = Color.white;
  75. [HideInInspector][SerializeField] Color mGradientBottom = new Color(0.7f, 0.7f, 0.7f);
  76. [HideInInspector][SerializeField] int mSpacingX = 0;
  77. [HideInInspector][SerializeField] int mSpacingY = 0;
  78. [HideInInspector][SerializeField] bool mUseFloatSpacing = false;
  79. [HideInInspector][SerializeField] float mFloatSpacingX = 0;
  80. [HideInInspector][SerializeField] float mFloatSpacingY = 0;
  81. [HideInInspector][SerializeField] bool mOverflowEllipsis = false;
  82. [HideInInspector][SerializeField] int mOverflowWidth = 0;
  83. [HideInInspector][SerializeField] Modifier mModifier = Modifier.None;
  84.  
  85. // Obsolete values
  86. [HideInInspector][SerializeField] bool mShrinkToFit = false;
  87. [HideInInspector][SerializeField] int mMaxLineWidth = 0;
  88. [HideInInspector][SerializeField] int mMaxLineHeight = 0;
  89. [HideInInspector][SerializeField] float mLineWidth = 0;
  90. [HideInInspector][SerializeField] bool mMultiline = true;
  91.  
  92. [System.NonSerialized] Font mActiveTTF = null;
  93. [System.NonSerialized] float mDensity = 1f;
  94. [System.NonSerialized] bool mShouldBeProcessed = true;
  95. [System.NonSerialized] string mProcessedText = null;
  96. [System.NonSerialized] bool mPremultiply = false;
  97. [System.NonSerialized] Vector2 mCalculatedSize = Vector2.zero;
  98. [System.NonSerialized] float mScale = 1f;
  99. [System.NonSerialized] int mFinalFontSize = 0;
  100. [System.NonSerialized] int mLastWidth = 0;
  101. [System.NonSerialized] int mLastHeight = 0;
  102.  
  103. /// <summary>
  104. /// Font size after modifications got taken into consideration such as shrinking content.
  105. /// </summary>
  106.  
  107. public int finalFontSize
  108. {
  109. get
  110. {
  111. if (trueTypeFont) return Mathf.RoundToInt(mScale * mFinalFontSize);
  112. return Mathf.RoundToInt(mFinalFontSize * mScale);
  113. }
  114. }
  115.  
  116. /// <summary>
  117. /// Function used to determine if something has changed (and thus the geometry must be rebuilt)
  118. /// </summary>
  119.  
  120. bool shouldBeProcessed
  121. {
  122. get
  123. {
  124. return mShouldBeProcessed;
  125. }
  126. set
  127. {
  128. if (value)
  129. {
  130. mChanged = true;
  131. mShouldBeProcessed = true;
  132. }
  133. else
  134. {
  135. mShouldBeProcessed = false;
  136. }
  137. }
  138. }
  139.  
  140. /// <summary>
  141. /// Whether the rectangle is anchored horizontally.
  142. /// </summary>
  143.  
  144. public override bool isAnchoredHorizontally { get { return base.isAnchoredHorizontally || mOverflow == Overflow.ResizeFreely; } }
  145.  
  146. /// <summary>
  147. /// Whether the rectangle is anchored vertically.
  148. /// </summary>
  149.  
  150. public override bool isAnchoredVertically
  151. {
  152. get
  153. {
  154. return base.isAnchoredVertically ||
  155. mOverflow == Overflow.ResizeFreely ||
  156. mOverflow == Overflow.ResizeHeight;
  157. }
  158. }
  159.  
  160. /// <summary>
  161. /// Retrieve the material used by the font.
  162. /// </summary>
  163.  
  164. public override Material material
  165. {
  166. get
  167. {
  168. if (mMat != null) return mMat;
  169. if (mFont != null) return mFont.material;
  170. if (mTrueTypeFont != null) return mTrueTypeFont.material;
  171. return null;
  172. }
  173. set
  174. {
  175. base.material = value;
  176. }
  177. }
  178.  
  179. /// <summary>
  180. /// Label's main texture comes from the font itself.
  181. /// </summary>
  182.  
  183. public override Texture mainTexture
  184. {
  185. get
  186. {
  187. if (mFont != null) return mFont.texture;
  188. if (mTrueTypeFont != null) { var mat = mTrueTypeFont.material; if (mat != null) return mat.mainTexture; }
  189. return null;
  190. }
  191. set
  192. {
  193. base.mainTexture = value;
  194. }
  195. }
  196.  
  197. [Obsolete("Use UILabel.bitmapFont instead")]
  198. public UIFont font { get { return bitmapFont; } set { bitmapFont = value; } }
  199.  
  200. /// <summary>
  201. /// Set the font used by this label.
  202. /// </summary>
  203.  
  204. public UIFont bitmapFont
  205. {
  206. get
  207. {
  208. return mFont;
  209. }
  210. set
  211. {
  212. if (mFont != value)
  213. {
  214. RemoveFromPanel();
  215. mFont = value;
  216. mTrueTypeFont = null;
  217. MarkAsChanged();
  218. }
  219. }
  220. }
  221.  
  222. /// <summary>
  223. /// Set the font used by this label.
  224. /// </summary>
  225.  
  226. public Font trueTypeFont
  227. {
  228. get
  229. {
  230. if (mTrueTypeFont != null) return mTrueTypeFont;
  231. return (mFont != null ? mFont.dynamicFont : null);
  232. }
  233. set
  234. {
  235. if (mTrueTypeFont != value)
  236. {
  237. SetActiveFont(null);
  238. RemoveFromPanel();
  239. mTrueTypeFont = value;
  240. shouldBeProcessed = true;
  241. mFont = null;
  242. SetActiveFont(value);
  243. ProcessAndRequest();
  244. if (mActiveTTF != null)
  245. base.MarkAsChanged();
  246. }
  247. }
  248. }
  249.  
  250. /// <summary>
  251. /// Ambiguous helper function.
  252. /// </summary>
  253.  
  254. public UnityEngine.Object ambigiousFont
  255. {
  256. get
  257. {
  258. return (UnityEngine.Object)mFont ?? (UnityEngine.Object)mTrueTypeFont;
  259. }
  260. set
  261. {
  262. UIFont bf = value as UIFont;
  263. if (bf != null) bitmapFont = bf;
  264. else trueTypeFont = value as Font;
  265. }
  266. }
  267.  
  268. /// <summary>
  269. /// Text that's being displayed by the label.
  270. /// </summary>
  271.  
  272. public string text
  273. {
  274. get
  275. {
  276. return mText;
  277. }
  278. set
  279. {
  280. if (mText == value) return;
  281.  
  282. if (string.IsNullOrEmpty(value))
  283. {
  284. if (!string.IsNullOrEmpty(mText))
  285. {
  286. mText = "";
  287. MarkAsChanged();
  288. ProcessAndRequest();
  289. }
  290. }
  291. else if (mText != value)
  292. {
  293. mText = value;
  294. MarkAsChanged();
  295. ProcessAndRequest();
  296. }
  297.  
  298. if (autoResizeBoxCollider) ResizeCollider();
  299. }
  300. }
  301.  
  302. /// <summary>
  303. /// Default font size.
  304. /// </summary>
  305.  
  306. public int defaultFontSize { get { return (trueTypeFont != null) ? mFontSize : (mFont != null ? mFont.defaultSize : 16); } }
  307.  
  308. /// <summary>
  309. /// Active font size used by the label.
  310. /// </summary>
  311.  
  312. public int fontSize
  313. {
  314. get
  315. {
  316. return mFontSize;
  317. }
  318. set
  319. {
  320. value = Mathf.Clamp(value, 0, 256);
  321.  
  322. if (mFontSize != value)
  323. {
  324. mFontSize = value;
  325. shouldBeProcessed = true;
  326. ProcessAndRequest();
  327. }
  328. }
  329. }
  330.  
  331. /// <summary>
  332. /// Dynamic font style used by the label.
  333. /// </summary>
  334.  
  335. public FontStyle fontStyle
  336. {
  337. get
  338. {
  339. return mFontStyle;
  340. }
  341. set
  342. {
  343. if (mFontStyle != value)
  344. {
  345. mFontStyle = value;
  346. shouldBeProcessed = true;
  347. ProcessAndRequest();
  348. }
  349. }
  350. }
  351.  
  352. /// <summary>
  353. /// Text alignment option.
  354. /// </summary>
  355.  
  356. public Alignment alignment
  357. {
  358. get
  359. {
  360. return mAlignment;
  361. }
  362. set
  363. {
  364. if (mAlignment != value)
  365. {
  366. mAlignment = value;
  367. shouldBeProcessed = true;
  368. ProcessAndRequest();
  369. }
  370. }
  371. }
  372.  
  373. /// <summary>
  374. /// Whether a gradient will be applied.
  375. /// </summary>
  376.  
  377. public bool applyGradient
  378. {
  379. get
  380. {
  381. return mApplyGradient;
  382. }
  383. set
  384. {
  385. if (mApplyGradient != value)
  386. {
  387. mApplyGradient = value;
  388. MarkAsChanged();
  389. }
  390. }
  391. }
  392.  
  393. /// <summary>
  394. /// Top gradient color.
  395. /// </summary>
  396.  
  397. public Color gradientTop
  398. {
  399. get
  400. {
  401. return mGradientTop;
  402. }
  403. set
  404. {
  405. if (mGradientTop != value)
  406. {
  407. mGradientTop = value;
  408. if (mApplyGradient) MarkAsChanged();
  409. }
  410. }
  411. }
  412.  
  413. /// <summary>
  414. /// Bottom gradient color.
  415. /// </summary>
  416.  
  417. public Color gradientBottom
  418. {
  419. get
  420. {
  421. return mGradientBottom;
  422. }
  423. set
  424. {
  425. if (mGradientBottom != value)
  426. {
  427. mGradientBottom = value;
  428. if (mApplyGradient) MarkAsChanged();
  429. }
  430. }
  431. }
  432.  
  433. /// <summary>
  434. /// Additional horizontal spacing between characters when printing text.
  435. /// </summary>
  436.  
  437. public int spacingX
  438. {
  439. get
  440. {
  441. return mSpacingX;
  442. }
  443. set
  444. {
  445. if (mSpacingX != value)
  446. {
  447. mSpacingX = value;
  448. MarkAsChanged();
  449. }
  450. }
  451. }
  452.  
  453. /// <summary>
  454. /// Additional vertical spacing between lines when printing text.
  455. /// </summary>
  456.  
  457. public int spacingY
  458. {
  459. get
  460. {
  461. return mSpacingY;
  462. }
  463. set
  464. {
  465. if (mSpacingY != value)
  466. {
  467. mSpacingY = value;
  468. MarkAsChanged();
  469. }
  470. }
  471. }
  472.  
  473. /// <summary>
  474. /// Whether this label will use float text spacing values, instead of integers.
  475. /// </summary>
  476.  
  477. public bool useFloatSpacing
  478. {
  479. get
  480. {
  481. return mUseFloatSpacing;
  482. }
  483. set
  484. {
  485. if (mUseFloatSpacing != value)
  486. {
  487. mUseFloatSpacing = value;
  488. shouldBeProcessed = true;
  489. }
  490. }
  491. }
  492.  
  493. /// <summary>
  494. /// Additional horizontal spacing between characters when printing text.
  495. /// For this to have any effect useFloatSpacing must be true.
  496. /// </summary>
  497.  
  498. public float floatSpacingX
  499. {
  500. get
  501. {
  502. return mFloatSpacingX;
  503. }
  504. set
  505. {
  506. if (!Mathf.Approximately(mFloatSpacingX, value))
  507. {
  508. mFloatSpacingX = value;
  509. MarkAsChanged();
  510. }
  511. }
  512. }
  513.  
  514. /// <summary>
  515. /// Additional vertical spacing between lines when printing text.
  516. /// For this to have any effect useFloatSpacing must be true.
  517. /// </summary>
  518.  
  519. public float floatSpacingY
  520. {
  521. get
  522. {
  523. return mFloatSpacingY;
  524. }
  525. set
  526. {
  527. if (!Mathf.Approximately(mFloatSpacingY, value))
  528. {
  529. mFloatSpacingY = value;
  530. MarkAsChanged();
  531. }
  532. }
  533. }
  534.  
  535. /// <summary>
  536. /// Convenience property to get the used y spacing.
  537. /// </summary>
  538.  
  539. public float effectiveSpacingY
  540. {
  541. get
  542. {
  543. return mUseFloatSpacing ? mFloatSpacingY : mSpacingY;
  544. }
  545. }
  546.  
  547. /// <summary>
  548. /// Convenience property to get the used x spacing.
  549. /// </summary>
  550.  
  551. public float effectiveSpacingX
  552. {
  553. get
  554. {
  555. return mUseFloatSpacing ? mFloatSpacingX : mSpacingX;
  556. }
  557. }
  558.  
  559. /// <summary>
  560. /// Whether to append "..." at the end of clamped text that didn't fit.
  561. /// </summary>
  562.  
  563. public bool overflowEllipsis
  564. {
  565. get
  566. {
  567. return mOverflowEllipsis;
  568. }
  569. set
  570. {
  571. if (mOverflowEllipsis != value)
  572. {
  573. mOverflowEllipsis = value;
  574. MarkAsChanged();
  575. }
  576. }
  577. }
  578.  
  579. /// <summary>
  580. /// Maximum width used when Resize Freely overflow type is enabled.
  581. /// If the printed text exceeds this width, it will wrap onto the following line.
  582. /// </summary>
  583.  
  584. public int overflowWidth
  585. {
  586. get
  587. {
  588. return mOverflowWidth;
  589. }
  590. set
  591. {
  592. if (mOverflowWidth != value)
  593. {
  594. mOverflowWidth = value;
  595. MarkAsChanged();
  596. }
  597. }
  598. }
  599.  
  600. /// <summary>
  601. /// Whether the label will use the printed size instead of font size when printing the label.
  602. /// It's a dynamic font feature that will ensure that the text is crisp when shrunk.
  603. /// </summary>
  604.  
  605. bool keepCrisp
  606. {
  607. get
  608. {
  609. if (trueTypeFont != null && keepCrispWhenShrunk != Crispness.Never)
  610. {
  611. #if UNITY_IPHONE || UNITY_ANDROID || UNITY_WP8 || UNITY_WP_8_1 || UNITY_BLACKBERRY
  612. return (keepCrispWhenShrunk == Crispness.Always);
  613. #else
  614. return true;
  615. #endif
  616. }
  617. return false;
  618. }
  619. }
  620.  
  621. /// <summary>
  622. /// Whether this label will support color encoding in the format of [RRGGBB] and new line in the form of a "\\n" string.
  623. /// </summary>
  624.  
  625. public bool supportEncoding
  626. {
  627. get
  628. {
  629. return mEncoding;
  630. }
  631. set
  632. {
  633. if (mEncoding != value)
  634. {
  635. mEncoding = value;
  636. shouldBeProcessed = true;
  637. }
  638. }
  639. }
  640.  
  641. /// <summary>
  642. /// Style used for symbols.
  643. /// </summary>
  644.  
  645. public NGUIText.SymbolStyle symbolStyle
  646. {
  647. get
  648. {
  649. return mSymbols;
  650. }
  651. set
  652. {
  653. if (mSymbols != value)
  654. {
  655. mSymbols = value;
  656. shouldBeProcessed = true;
  657. }
  658. }
  659. }
  660.  
  661. /// <summary>
  662. /// Overflow method controls the label's behaviour when its content doesn't fit the bounds.
  663. /// </summary>
  664.  
  665. public Overflow overflowMethod
  666. {
  667. get
  668. {
  669. return mOverflow;
  670. }
  671. set
  672. {
  673. if (mOverflow != value)
  674. {
  675. mOverflow = value;
  676. shouldBeProcessed = true;
  677. }
  678. }
  679. }
  680.  
  681. /// <summary>
  682. /// Maximum width of the label in pixels.
  683. /// </summary>
  684.  
  685. [System.Obsolete("Use 'width' instead")]
  686. public int lineWidth
  687. {
  688. get { return width; }
  689. set { width = value; }
  690. }
  691.  
  692. /// <summary>
  693. /// Maximum height of the label in pixels.
  694. /// </summary>
  695.  
  696. [System.Obsolete("Use 'height' instead")]
  697. public int lineHeight
  698. {
  699. get { return height; }
  700. set { height = value; }
  701. }
  702.  
  703. /// <summary>
  704. /// Whether the label supports multiple lines.
  705. /// </summary>
  706. public bool multiLine
  707. {
  708. get
  709. {
  710. return mMaxLineCount != 1;
  711. }
  712. set
  713. {
  714. if ((mMaxLineCount != 1) != value)
  715. {
  716. mMaxLineCount = (value ? 0 : 1);
  717. shouldBeProcessed = true;
  718. }
  719. }
  720. }
  721.  
  722. /// <summary>
  723. /// Process the label's text before returning its corners.
  724. /// </summary>
  725.  
  726. public override Vector3[] localCorners
  727. {
  728. get
  729. {
  730. if (shouldBeProcessed) ProcessText();
  731. return base.localCorners;
  732. }
  733. }
  734.  
  735. /// <summary>
  736. /// Process the label's text before returning its corners.
  737. /// </summary>
  738.  
  739. public override Vector3[] worldCorners
  740. {
  741. get
  742. {
  743. if (shouldBeProcessed) ProcessText();
  744. return base.worldCorners;
  745. }
  746. }
  747.  
  748. /// <summary>
  749. /// Process the label's text before returning its drawing dimensions.
  750. /// </summary>
  751.  
  752. public override Vector4 drawingDimensions
  753. {
  754. get
  755. {
  756. if (shouldBeProcessed) ProcessText();
  757. return base.drawingDimensions;
  758. }
  759. }
  760.  
  761. /// <summary>
  762. /// The max number of lines to be displayed for the label
  763. /// </summary>
  764.  
  765. public int maxLineCount
  766. {
  767. get
  768. {
  769. return mMaxLineCount;
  770. }
  771. set
  772. {
  773. if (mMaxLineCount != value)
  774. {
  775. mMaxLineCount = Mathf.Max(value, 0);
  776. shouldBeProcessed = true;
  777. if (overflowMethod == Overflow.ShrinkContent) MakePixelPerfect();
  778. }
  779. }
  780. }
  781.  
  782. /// <summary>
  783. /// What effect is used by the label.
  784. /// </summary>
  785.  
  786. public Effect effectStyle
  787. {
  788. get
  789. {
  790. return mEffectStyle;
  791. }
  792. set
  793. {
  794. if (mEffectStyle != value)
  795. {
  796. mEffectStyle = value;
  797. shouldBeProcessed = true;
  798. }
  799. }
  800. }
  801.  
  802. /// <summary>
  803. /// Color used by the effect, if it's enabled.
  804. /// </summary>
  805.  
  806. public Color effectColor
  807. {
  808. get
  809. {
  810. return mEffectColor;
  811. }
  812. set
  813. {
  814. if (mEffectColor != value)
  815. {
  816. mEffectColor = value;
  817. if (mEffectStyle != Effect.None) shouldBeProcessed = true;
  818. }
  819. }
  820. }
  821.  
  822. /// <summary>
  823. /// Effect distance in pixels.
  824. /// </summary>
  825.  
  826. public Vector2 effectDistance
  827. {
  828. get
  829. {
  830. return mEffectDistance;
  831. }
  832. set
  833. {
  834. if (mEffectDistance != value)
  835. {
  836. mEffectDistance = value;
  837. shouldBeProcessed = true;
  838. }
  839. }
  840. }
  841.  
  842. /// <summary>
  843. /// How many quads there are per printed character.
  844. /// </summary>
  845. public int quadsPerCharacter
  846. {
  847. get
  848. {
  849. if (mEffectStyle == Effect.Shadow) return 2;
  850. else if (mEffectStyle == Effect.Outline) return 5;
  851. else if (mEffectStyle == Effect.Outline8) return 9;
  852. return 1;
  853. }
  854. }
  855.  
  856. /// <summary>
  857. /// Whether the label will automatically shrink its size in order to fit the maximum line width.
  858. /// </summary>
  859.  
  860. [System.Obsolete("Use 'overflowMethod == UILabel.Overflow.ShrinkContent' instead")]
  861. public bool shrinkToFit
  862. {
  863. get
  864. {
  865. return mOverflow == Overflow.ShrinkContent;
  866. }
  867. set
  868. {
  869. if (value)
  870. {
  871. overflowMethod = Overflow.ShrinkContent;
  872. }
  873. }
  874. }
  875.  
  876. /// <summary>
  877. /// Returns the processed version of 'text', with new line characters, line wrapping, etc.
  878. /// </summary>
  879.  
  880. public string processedText
  881. {
  882. get
  883. {
  884. if (mLastWidth != mWidth || mLastHeight != mHeight)
  885. {
  886. mLastWidth = mWidth;
  887. mLastHeight = mHeight;
  888. mShouldBeProcessed = true;
  889. }
  890.  
  891. // Process the text if necessary
  892. if (shouldBeProcessed) ProcessText();
  893. return mProcessedText;
  894. }
  895. }
  896.  
  897. /// <summary>
  898. /// Actual printed size of the text, in pixels.
  899. /// </summary>
  900.  
  901. public Vector2 printedSize
  902. {
  903. get
  904. {
  905. if (shouldBeProcessed) ProcessText();
  906. return mCalculatedSize;
  907. }
  908. }
  909.  
  910. /// <summary>
  911. /// Local size of the widget, in pixels.
  912. /// </summary>
  913.  
  914. public override Vector2 localSize
  915. {
  916. get
  917. {
  918. if (shouldBeProcessed) ProcessText();
  919. return base.localSize;
  920. }
  921. }
  922.  
  923. /// <summary>
  924. /// Whether the label has a valid font.
  925. /// </summary>
  926.  
  927. bool isValid { get { return mFont != null || mTrueTypeFont != null; } }
  928.  
  929. /// <summary>
  930. /// Custom text modifier that can transform the visible text when the label's text is assigned.
  931. /// </summary>
  932.  
  933. public ModifierFunc customModifier;
  934. public delegate string ModifierFunc (string s);
  935.  
  936. /// <summary>
  937. /// Text modifier can transform the text that's actually printed, without altering the label's text value.
  938. /// </summary>
  939.  
  940. public Modifier modifier
  941. {
  942. get
  943. {
  944. return mModifier;
  945. }
  946. set
  947. {
  948. if (mModifier != value)
  949. {
  950. mModifier = value;
  951. MarkAsChanged();
  952. ProcessAndRequest();
  953. }
  954. }
  955. }
  956.  
  957. static BetterList<UILabel> mList = new BetterList<UILabel>();
  958. static Dictionary<Font, int> mFontUsage = new Dictionary<Font, int>();
  959.  
  960. /// <summary>
  961. /// Register the font texture change listener.
  962. /// </summary>
  963.  
  964. protected override void OnInit ()
  965. {
  966. base.OnInit();
  967. mList.Add(this);
  968. SetActiveFont(trueTypeFont);
  969. }
  970.  
  971. /// <summary>
  972. /// Remove the font texture change listener.
  973. /// </summary>
  974.  
  975. protected override void OnDisable ()
  976. {
  977. SetActiveFont(null);
  978. mList.Remove(this);
  979. base.OnDisable();
  980. }
  981.  
  982. /// <summary>
  983. /// Set the active font, correctly setting and clearing callbacks.
  984. /// </summary>
  985.  
  986. protected void SetActiveFont (Font fnt)
  987. {
  988. if (mActiveTTF != fnt)
  989. {
  990. Font font = mActiveTTF;
  991.  
  992. if (font != null)
  993. {
  994. int usage;
  995.  
  996. if (mFontUsage.TryGetValue(font, out usage))
  997. {
  998. usage = Mathf.Max(0, --usage);
  999.  
  1000. if (usage == 0)
  1001. {
  1002. #if UNITY_4_3 || UNITY_4_5
  1003. font.textureRebuildCallback = null;
  1004. #endif
  1005. mFontUsage.Remove(font);
  1006. }
  1007. else mFontUsage[font] = usage;
  1008. }
  1009. #if UNITY_4_3 || UNITY_4_5
  1010. else font.textureRebuildCallback = null;
  1011. #endif
  1012. }
  1013.  
  1014. mActiveTTF = fnt;
  1015. font = fnt;
  1016.  
  1017. if (font != null)
  1018. {
  1019. int usage = 0;
  1020.  
  1021. // Font hasn't been used yet? Register a change delegate callback
  1022. #if UNITY_4_3 || UNITY_4_5
  1023. if (!mFontUsage.TryGetValue(font, out usage))
  1024. font.textureRebuildCallback = delegate() { OnFontChanged(font); };
  1025. #endif
  1026. #if UNITY_FLASH
  1027. mFontUsage[font] = usage + 1;
  1028. #else
  1029. mFontUsage[font] = ++usage;
  1030. #endif
  1031. }
  1032. }
  1033. }
  1034.  
  1035. /// <summary>
  1036. /// Label's actual printed text may be modified before being drawn.
  1037. /// </summary>
  1038.  
  1039. public string printedText
  1040. {
  1041. get
  1042. {
  1043. if (!string.IsNullOrEmpty(mText))
  1044. {
  1045. if (mModifier == Modifier.None) return mText;
  1046. if (mModifier == Modifier.ToLowercase) return mText.ToLower();
  1047. if (mModifier == Modifier.ToUppercase) return mText.ToUpper();
  1048. if (mModifier == Modifier.Custom && customModifier != null) return customModifier(mText);
  1049. }
  1050. return mText;
  1051. }
  1052. }
  1053.  
  1054. /// <summary>
  1055. /// Notification called when the Unity's font's texture gets rebuilt.
  1056. /// Unity's font has a nice tendency to simply discard other characters when the texture's dimensions change.
  1057. /// By requesting them inside the notification callback, we immediately force them back in.
  1058. /// Originally I was subscribing each label to the font individually, but as it turned out
  1059. /// mono's delegate system causes an insane amount of memory allocations when += or -= to a delegate.
  1060. /// So... queue yet another work-around.
  1061. /// </summary>
  1062.  
  1063. static void OnFontChanged (Font font)
  1064. {
  1065. for (int i = 0; i < mList.size; ++i)
  1066. {
  1067. UILabel lbl = mList[i];
  1068.  
  1069. if (lbl != null)
  1070. {
  1071. Font fnt = lbl.trueTypeFont;
  1072.  
  1073. if (fnt == font)
  1074. {
  1075. fnt.RequestCharactersInTexture(lbl.mText, lbl.mFinalFontSize, lbl.mFontStyle);
  1076. lbl.MarkAsChanged();
  1077.  
  1078. if (lbl.panel == null)
  1079. lbl.CreatePanel();
  1080.  
  1081. if (mTempDrawcalls == null)
  1082. mTempDrawcalls = new BetterList<UIDrawCall>();
  1083.  
  1084. if (lbl.drawCall != null && !mTempDrawcalls.Contains(lbl.drawCall))
  1085. mTempDrawcalls.Add(lbl.drawCall);
  1086. }
  1087. }
  1088. }
  1089.  
  1090. if (mTempDrawcalls != null)
  1091. {
  1092. for (int i = 0, imax = mTempDrawcalls.size; i < imax; ++i)
  1093. {
  1094. UIDrawCall dc = mTempDrawcalls[i];
  1095. if (dc.panel != null) dc.panel.FillDrawCall(dc);
  1096. }
  1097. mTempDrawcalls.Clear();
  1098. }
  1099.  
  1100. // http://www.tasharen.com/forum/index.php?topic=15202.0
  1101. /*++mFontChangedDepth;
  1102.  
  1103. for (int i = 0; i < mList.size; ++i)
  1104. {
  1105. UILabel lbl = mList[i];
  1106.  
  1107. if (lbl != null)v [kp;' 5rfc328i0u
  1108. {
  1109. Font fnt = lbl.trueTypeFont;
  1110.  
  1111. if (fnt == font)
  1112. {
  1113. if (mTempLabels == null) mTempLabels = new BetterList<UILabel>();
  1114.  
  1115. if (!mTempLabels.Contains(lbl))
  1116. {
  1117. mTempLabels.Add(lbl);
  1118. lbl.UpdateNGUIText();
  1119. fnt.RequestCharactersInTexture(lbl.printedText, lbl.mFinalFontSize, lbl.mFontStyle);
  1120. }
  1121. }
  1122. }
  1123. }
  1124.  
  1125. if (mFontChangedDepth == 1 && mTempLabels != null)
  1126. {
  1127. var frame = Time.frameCount;
  1128.  
  1129. for (int i = 0, imax = mTempLabels.size; i < imax; ++i)
  1130. {
  1131. var lbl = mTempLabels.buffer[i];
  1132. if (lbl.panel == null) lbl.CreatePanel();
  1133. lbl.MarkAsChanged();
  1134. lbl.UpdateGeometry(frame);
  1135.  
  1136. var dc = lbl.drawCall;
  1137.  
  1138. if (dc != null)
  1139. {
  1140. if (mTempDrawcalls == null) mTempDrawcalls = new BetterList<UIDrawCall>();
  1141. if (!mTempDrawcalls.Contains(dc)) mTempDrawcalls.Add(dc);
  1142. }
  1143. }
  1144.  
  1145. mTempLabels.Clear();
  1146.  
  1147. if (mTempDrawcalls != null)
  1148. {
  1149. for (int i = 0, imax = mTempDrawcalls.size; i < imax; ++i)
  1150. {
  1151. var dc = mTempDrawcalls.buffer[i];
  1152. if (dc.panel != null) dc.panel.FillDrawCall(dc);
  1153. }
  1154. mTempDrawcalls.Clear();
  1155. }
  1156. }
  1157. --mFontChangedDepth;*/
  1158. }
  1159.  
  1160. //[System.NonSerialized] static int mFontChangedDepth = 0;
  1161. [System.NonSerialized] static BetterList<UIDrawCall> mTempDrawcalls;
  1162. //[System.NonSerialized] static BetterList<UILabel> mTempLabels;
  1163.  
  1164. /// <summary>
  1165. /// Get the sides of the rectangle relative to the specified transform.
  1166. /// The order is left, top, right, bottom.
  1167. /// </summary>
  1168.  
  1169. public override Vector3[] GetSides (Transform relativeTo)
  1170. {
  1171. if (shouldBeProcessed) ProcessText();
  1172. return base.GetSides(relativeTo);
  1173. }
  1174.  
  1175. /// <summary>
  1176. /// Upgrading labels is a bit different.
  1177. /// </summary>
  1178.  
  1179. protected override void UpgradeFrom265 ()
  1180. {
  1181. ProcessText(true, true);
  1182.  
  1183. if (mShrinkToFit)
  1184. {
  1185. overflowMethod = Overflow.ShrinkContent;
  1186. mMaxLineCount = 0;
  1187. }
  1188.  
  1189. if (mMaxLineWidth != 0)
  1190. {
  1191. width = mMaxLineWidth;
  1192. overflowMethod = mMaxLineCount > 0 ? Overflow.ResizeHeight : Overflow.ShrinkContent;
  1193. }
  1194. else overflowMethod = Overflow.ResizeFreely;
  1195.  
  1196. if (mMaxLineHeight != 0)
  1197. height = mMaxLineHeight;
  1198.  
  1199. if (mFont != null)
  1200. {
  1201. int min = mFont.defaultSize;
  1202. if (height < min) height = min;
  1203. fontSize = min;
  1204. }
  1205.  
  1206. mMaxLineWidth = 0;
  1207. mMaxLineHeight = 0;
  1208. mShrinkToFit = false;
  1209.  
  1210. NGUITools.UpdateWidgetCollider(gameObject, true);
  1211. }
  1212.  
  1213. /// <summary>
  1214. /// If the label is anchored it should not auto-resize.
  1215. /// </summary>
  1216.  
  1217. protected override void OnAnchor ()
  1218. {
  1219. if (mOverflow == Overflow.ResizeFreely)
  1220. {
  1221. if (isFullyAnchored)
  1222. mOverflow = Overflow.ShrinkContent;
  1223. }
  1224. else if (mOverflow == Overflow.ResizeHeight)
  1225. {
  1226. if (topAnchor.target != null && bottomAnchor.target != null)
  1227. mOverflow = Overflow.ShrinkContent;
  1228. }
  1229. base.OnAnchor();
  1230. }
  1231.  
  1232. /// <summary>
  1233. /// Request the needed characters in the texture.
  1234. /// </summary>
  1235.  
  1236. void ProcessAndRequest ()
  1237. {
  1238. #if UNITY_EDITOR
  1239. if (!Application.isPlaying && !NGUITools.GetActive(this)) return;
  1240. if (!mAllowProcessing) return;
  1241. #endif
  1242. if (ambigiousFont != null) ProcessText();
  1243. }
  1244.  
  1245. #if UNITY_EDITOR
  1246. // Used to ensure that we don't process font more than once inside OnValidate function below
  1247. [System.NonSerialized] bool mAllowProcessing = true;
  1248. [System.NonSerialized] bool mUsingTTF = true;
  1249.  
  1250. /// <summary>
  1251. /// Validate the properties.
  1252. /// </summary>
  1253.  
  1254. protected override void OnValidate ()
  1255. {
  1256. base.OnValidate();
  1257.  
  1258. if (NGUITools.GetActive(this))
  1259. {
  1260. Font ttf = mTrueTypeFont;
  1261. UIFont fnt = mFont;
  1262.  
  1263. // If the true type font was not used before, but now it is, clear the font reference
  1264. if (!mUsingTTF && ttf != null) fnt = null;
  1265. else if (mUsingTTF && fnt != null) ttf = null;
  1266.  
  1267. mFont = null;
  1268. mTrueTypeFont = null;
  1269. mAllowProcessing = false;
  1270. SetActiveFont(null);
  1271.  
  1272. if (fnt != null)
  1273. {
  1274. bitmapFont = fnt;
  1275. mUsingTTF = false;
  1276. }
  1277. else if (ttf != null)
  1278. {
  1279. trueTypeFont = ttf;
  1280. mUsingTTF = true;
  1281. }
  1282.  
  1283. shouldBeProcessed = true;
  1284. mAllowProcessing = true;
  1285. ProcessAndRequest();
  1286. if (autoResizeBoxCollider) ResizeCollider();
  1287. }
  1288. }
  1289. #endif
  1290.  
  1291. #if !UNITY_4_3 && !UNITY_4_5
  1292. static bool mTexRebuildAdded = false;
  1293.  
  1294. protected override void OnEnable ()
  1295. {
  1296. base.OnEnable();
  1297. if (!mTexRebuildAdded)
  1298. {
  1299. mTexRebuildAdded = true;
  1300. Font.textureRebuilt += OnFontChanged;
  1301. }
  1302. }
  1303. #endif
  1304.  
  1305. /// <summary>
  1306. /// Determine start-up values.
  1307. /// </summary>
  1308.  
  1309. protected override void OnStart ()
  1310. {
  1311. base.OnStart();
  1312.  
  1313. // Legacy support
  1314. if (mLineWidth > 0f)
  1315. {
  1316. mMaxLineWidth = Mathf.RoundToInt(mLineWidth);
  1317. mLineWidth = 0f;
  1318. }
  1319.  
  1320. if (!mMultiline)
  1321. {
  1322. mMaxLineCount = 1;
  1323. mMultiline = true;
  1324. }
  1325.  
  1326. // Whether this is a premultiplied alpha shader
  1327. mPremultiply = (material != null && material.shader != null && material.shader.name.Contains("Premultiplied"));
  1328.  
  1329. // Request the text within the font
  1330. ProcessAndRequest();
  1331. }
  1332.  
  1333. /// <summary>
  1334. /// UILabel needs additional processing when something changes.
  1335. /// </summary>
  1336.  
  1337. public override void MarkAsChanged ()
  1338. {
  1339. shouldBeProcessed = true;
  1340. base.MarkAsChanged();
  1341. }
  1342.  
  1343. /// <summary>
  1344. /// Process the raw text, called when something changes.
  1345. /// </summary>
  1346.  
  1347. public void ProcessText (bool legacyMode = false, bool full = true)
  1348. {
  1349. if (!isValid) return;
  1350.  
  1351. mChanged = true;
  1352. shouldBeProcessed = false;
  1353.  
  1354. float regionX = mDrawRegion.z - mDrawRegion.x;
  1355. float regionY = mDrawRegion.w - mDrawRegion.y;
  1356.  
  1357. NGUIText.rectWidth = legacyMode ? (mMaxLineWidth != 0 ? mMaxLineWidth : 1000000) : width;
  1358. NGUIText.rectHeight = legacyMode ? (mMaxLineHeight != 0 ? mMaxLineHeight : 1000000) : height;
  1359. NGUIText.regionWidth = (regionX != 1f) ? Mathf.RoundToInt(NGUIText.rectWidth * regionX) : NGUIText.rectWidth;
  1360. NGUIText.regionHeight = (regionY != 1f) ? Mathf.RoundToInt(NGUIText.rectHeight * regionY) : NGUIText.rectHeight;
  1361.  
  1362. mFinalFontSize = Mathf.Abs(legacyMode ? Mathf.RoundToInt(cachedTransform.localScale.x) : defaultFontSize);
  1363. mScale = 1f;
  1364.  
  1365. if (NGUIText.regionWidth < 1 || NGUIText.regionHeight < 0)
  1366. {
  1367. mProcessedText = "";
  1368. return;
  1369. }
  1370.  
  1371. bool isDynamic = (trueTypeFont != null);
  1372.  
  1373. if (isDynamic && keepCrisp)
  1374. {
  1375. UIRoot rt = root;
  1376. if (rt != null) mDensity = (rt != null) ? rt.pixelSizeAdjustment : 1f;
  1377. }
  1378. else mDensity = 1f;
  1379.  
  1380. if (full) UpdateNGUIText();
  1381.  
  1382. if (mOverflow == Overflow.ResizeFreely)
  1383. {
  1384. NGUIText.rectWidth = 1000000;
  1385. NGUIText.regionWidth = 1000000;
  1386.  
  1387. if (mOverflowWidth > 0)
  1388. {
  1389. NGUIText.rectWidth = Mathf.Min(NGUIText.rectWidth, mOverflowWidth);
  1390. NGUIText.regionWidth = Mathf.Min(NGUIText.regionWidth, mOverflowWidth);
  1391. }
  1392. }
  1393.  
  1394. if (mOverflow == Overflow.ResizeFreely || mOverflow == Overflow.ResizeHeight)
  1395. {
  1396. NGUIText.rectHeight = 1000000;
  1397. NGUIText.regionHeight = 1000000;
  1398. }
  1399.  
  1400. if (mFinalFontSize > 0)
  1401. {
  1402. bool adjustSize = keepCrisp;
  1403.  
  1404. for (int ps = mFinalFontSize; ps > 0; --ps)
  1405. {
  1406. // Adjust either the size, or the scale
  1407. if (adjustSize)
  1408. {
  1409. mFinalFontSize = ps;
  1410. NGUIText.fontSize = mFinalFontSize;
  1411. }
  1412. else
  1413. {
  1414. mScale = (float)ps / mFinalFontSize;
  1415. NGUIText.fontScale = isDynamic ? mScale : ((float)mFontSize / mFont.defaultSize) * mScale;
  1416. }
  1417.  
  1418. NGUIText.Update(false);
  1419.  
  1420. // Wrap the text
  1421. bool fits = NGUIText.WrapText(printedText, out mProcessedText, false, false, mOverflow == Overflow.ClampContent && mOverflowEllipsis);
  1422.  
  1423. if (mOverflow == Overflow.ShrinkContent && !fits)
  1424. {
  1425. if (--ps > 1) continue;
  1426. else break;
  1427. }
  1428. else if (mOverflow == Overflow.ResizeFreely)
  1429. {
  1430. mCalculatedSize = NGUIText.CalculatePrintedSize(mProcessedText);
  1431.  
  1432. int w = Mathf.Max(minWidth, Mathf.RoundToInt(mCalculatedSize.x));
  1433. if (regionX != 1f) w = Mathf.RoundToInt(w / regionX);
  1434. int h = Mathf.Max(minHeight, Mathf.RoundToInt(mCalculatedSize.y));
  1435. if (regionY != 1f) h = Mathf.RoundToInt(h / regionY);
  1436.  
  1437. if ((w & 1) == 1) ++w;
  1438. if ((h & 1) == 1) ++h;
  1439.  
  1440. if (mWidth != w || mHeight != h)
  1441. {
  1442. mWidth = w;
  1443. mHeight = h;
  1444. if (onChange != null) onChange();
  1445. }
  1446. }
  1447. else if (mOverflow == Overflow.ResizeHeight)
  1448. {
  1449. mCalculatedSize = NGUIText.CalculatePrintedSize(mProcessedText);
  1450. int h = Mathf.Max(minHeight, Mathf.RoundToInt(mCalculatedSize.y));
  1451. if (regionY != 1f) h = Mathf.RoundToInt(h / regionY);
  1452. if ((h & 1) == 1) ++h;
  1453.  
  1454. if (mHeight != h)
  1455. {
  1456. mHeight = h;
  1457. if (onChange != null) onChange();
  1458. }
  1459. }
  1460. else
  1461. {
  1462. mCalculatedSize = NGUIText.CalculatePrintedSize(mProcessedText);
  1463. }
  1464.  
  1465. // Upgrade to the new system
  1466. if (legacyMode)
  1467. {
  1468. width = Mathf.RoundToInt(mCalculatedSize.x);
  1469. height = Mathf.RoundToInt(mCalculatedSize.y);
  1470. cachedTransform.localScale = Vector3.one;
  1471. }
  1472. break;
  1473. }
  1474. }
  1475. else
  1476. {
  1477. cachedTransform.localScale = Vector3.one;
  1478. mProcessedText = "";
  1479. mScale = 1f;
  1480. }
  1481. if (full)
  1482. {
  1483. NGUIText.bitmapFont = null;
  1484. NGUIText.dynamicFont = null;
  1485. }
  1486. }
  1487.  
  1488. /// <summary>
  1489. /// Text is pixel-perfect when its scale matches the size.
  1490. /// </summary>
  1491.  
  1492. public override void MakePixelPerfect ()
  1493. {
  1494. if (ambigiousFont != null)
  1495. {
  1496. Vector3 pos = cachedTransform.localPosition;
  1497. pos.x = Mathf.RoundToInt(pos.x);
  1498. pos.y = Mathf.RoundToInt(pos.y);
  1499. pos.z = Mathf.RoundToInt(pos.z);
  1500.  
  1501. cachedTransform.localPosition = pos;
  1502. cachedTransform.localScale = Vector3.one;
  1503.  
  1504. if (mOverflow == Overflow.ResizeFreely)
  1505. {
  1506. AssumeNaturalSize();
  1507. }
  1508. else
  1509. {
  1510. int w = width;
  1511. int h = height;
  1512.  
  1513. Overflow over = mOverflow;
  1514. if (over != Overflow.ResizeHeight) mWidth = 100000;
  1515. mHeight = 100000;
  1516.  
  1517. mOverflow = Overflow.ShrinkContent;
  1518. ProcessText(false, true);
  1519. mOverflow = over;
  1520.  
  1521. int minX = Mathf.RoundToInt(mCalculatedSize.x);
  1522. int minY = Mathf.RoundToInt(mCalculatedSize.y);
  1523.  
  1524. minX = Mathf.Max(minX, base.minWidth);
  1525. minY = Mathf.Max(minY, base.minHeight);
  1526.  
  1527. if ((minX & 1) == 1) ++minX;
  1528. if ((minY & 1) == 1) ++minY;
  1529.  
  1530. mWidth = Mathf.Max(w, minX);
  1531. mHeight = Mathf.Max(h, minY);
  1532.  
  1533. MarkAsChanged();
  1534. }
  1535. }
  1536. else base.MakePixelPerfect();
  1537. }
  1538.  
  1539. /// <summary>
  1540. /// Make the label assume its natural size.
  1541. /// </summary>
  1542.  
  1543. public void AssumeNaturalSize ()
  1544. {
  1545. if (ambigiousFont != null)
  1546. {
  1547. mWidth = 100000;
  1548. mHeight = 100000;
  1549. ProcessText(false, true);
  1550. mWidth = Mathf.RoundToInt(mCalculatedSize.x);
  1551. mHeight = Mathf.RoundToInt(mCalculatedSize.y);
  1552. if ((mWidth & 1) == 1) ++mWidth;
  1553. if ((mHeight & 1) == 1) ++mHeight;
  1554. MarkAsChanged();
  1555. }
  1556. }
  1557.  
  1558. [System.Obsolete("Use UILabel.GetCharacterAtPosition instead")]
  1559. public int GetCharacterIndex (Vector3 worldPos) { return GetCharacterIndexAtPosition(worldPos, false); }
  1560.  
  1561. [System.Obsolete("Use UILabel.GetCharacterAtPosition instead")]
  1562. public int GetCharacterIndex (Vector2 localPos) { return GetCharacterIndexAtPosition(localPos, false); }
  1563.  
  1564. static List<Vector3> mTempVerts = new List<Vector3>();
  1565. static List<int> mTempIndices = new List<int>();
  1566.  
  1567. /// <summary>
  1568. /// Return the index of the character at the specified world position.
  1569. /// </summary>
  1570.  
  1571. public int GetCharacterIndexAtPosition (Vector3 worldPos, bool precise)
  1572. {
  1573. Vector2 localPos = cachedTransform.InverseTransformPoint(worldPos);
  1574. return GetCharacterIndexAtPosition(localPos, precise);
  1575. }
  1576.  
  1577. /// <summary>
  1578. /// Return the index of the character at the specified local position.
  1579. /// </summary>
  1580.  
  1581. public int GetCharacterIndexAtPosition (Vector2 localPos, bool precise)
  1582. {
  1583. if (isValid)
  1584. {
  1585. string text = processedText;
  1586. if (string.IsNullOrEmpty(text)) return 0;
  1587.  
  1588. UpdateNGUIText();
  1589.  
  1590. if (precise) NGUIText.PrintExactCharacterPositions(text, mTempVerts, mTempIndices);
  1591. else NGUIText.PrintApproximateCharacterPositions(text, mTempVerts, mTempIndices);
  1592.  
  1593. if (mTempVerts.Count > 0)
  1594. {
  1595. ApplyOffset(mTempVerts, 0);
  1596. int retVal = precise ?
  1597. NGUIText.GetExactCharacterIndex(mTempVerts, mTempIndices, localPos) :
  1598. NGUIText.GetApproximateCharacterIndex(mTempVerts, mTempIndices, localPos);
  1599.  
  1600. mTempVerts.Clear();
  1601. mTempIndices.Clear();
  1602.  
  1603. NGUIText.bitmapFont = null;
  1604. NGUIText.dynamicFont = null;
  1605. return retVal;
  1606. }
  1607.  
  1608. NGUIText.bitmapFont = null;
  1609. NGUIText.dynamicFont = null;
  1610. }
  1611. return 0;
  1612. }
  1613.  
  1614. /// <summary>
  1615. /// Retrieve the word directly below the specified world-space position.
  1616. /// </summary>
  1617.  
  1618. public string GetWordAtPosition (Vector3 worldPos)
  1619. {
  1620. int index = GetCharacterIndexAtPosition(worldPos, true);
  1621. return GetWordAtCharacterIndex(index);
  1622. }
  1623.  
  1624. /// <summary>
  1625. /// Retrieve the word directly below the specified relative-to-label position.
  1626. /// </summary>
  1627.  
  1628. public string GetWordAtPosition (Vector2 localPos)
  1629. {
  1630. int index = GetCharacterIndexAtPosition(localPos, true);
  1631. return GetWordAtCharacterIndex(index);
  1632. }
  1633.  
  1634. /// <summary>
  1635. /// Retrieve the word right under the specified character index.
  1636. /// </summary>
  1637.  
  1638. public string GetWordAtCharacterIndex (int characterIndex)
  1639. {
  1640. string s = printedText;
  1641.  
  1642. if (characterIndex != -1 && characterIndex < s.Length)
  1643. {
  1644. #if UNITY_FLASH
  1645. int wordStart = LastIndexOfAny(s, new char[] { ' ', '\n' }, characterIndex) + 1;
  1646. int wordEnd = IndexOfAny(s, new char[] { ' ', '\n', ',', '.' }, characterIndex);
  1647. #else
  1648. int wordStart = s.LastIndexOfAny(new char[] {' ', '\n'}, characterIndex) + 1;
  1649. int wordEnd = s.IndexOfAny(new char[] { ' ', '\n', ',', '.' }, characterIndex);
  1650. #endif
  1651. if (wordEnd == -1) wordEnd = s.Length;
  1652.  
  1653. if (wordStart != wordEnd)
  1654. {
  1655. int len = wordEnd - wordStart;
  1656.  
  1657. if (len > 0)
  1658. {
  1659. string word = s.Substring(wordStart, len);
  1660. return NGUIText.StripSymbols(word);
  1661. }
  1662. }
  1663. }
  1664. return null;
  1665. }
  1666.  
  1667. #if UNITY_FLASH
  1668. /// <summary>
  1669. /// Flash is fail IRL: http://www.tasharen.com/forum/index.php?topic=11390.0
  1670. /// </summary>
  1671.  
  1672. int LastIndexOfAny (string input, char[] any, int start)
  1673. {
  1674. if (start >= 0 && input.Length > 0 && any.Length > 0 && start < input.Length)
  1675. {
  1676. for (int w = start; w >= 0; w--)
  1677. {
  1678. for (int r = 0; r < any.Length; r++)
  1679. {
  1680. if (any[r] == input[w])
  1681. {
  1682. return w;
  1683. }
  1684. }
  1685. }
  1686. }
  1687. return -1;
  1688. }
  1689.  
  1690. /// <summary>
  1691. /// Flash is fail IRL: http://www.tasharen.com/forum/index.php?topic=11390.0
  1692. /// </summary>
  1693.  
  1694. int IndexOfAny (string input, char[] any, int start)
  1695. {
  1696. if (start >= 0 && input.Length > 0 && any.Length > 0 && start < input.Length)
  1697. {
  1698. for (int w = start; w < input.Length; w++)
  1699. {
  1700. for (int r = 0; r < any.Length; r++)
  1701. {
  1702. if (any[r] == input[w])
  1703. {
  1704. return w;
  1705. }
  1706. }
  1707. }
  1708. }
  1709. return -1;
  1710. }
  1711. #endif
  1712.  
  1713. /// <summary>
  1714. /// Retrieve the URL directly below the specified world-space position.
  1715. /// </summary>
  1716.  
  1717. public string GetUrlAtPosition (Vector3 worldPos) { return GetUrlAtCharacterIndex(GetCharacterIndexAtPosition(worldPos, true)); }
  1718.  
  1719. /// <summary>
  1720. /// Retrieve the URL directly below the specified relative-to-label position.
  1721. /// </summary>
  1722.  
  1723. public string GetUrlAtPosition (Vector2 localPos) { return GetUrlAtCharacterIndex(GetCharacterIndexAtPosition(localPos, true)); }
  1724.  
  1725. /// <summary>
  1726. /// Retrieve the URL right under the specified character index.
  1727. /// </summary>
  1728.  
  1729. public string GetUrlAtCharacterIndex (int characterIndex)
  1730. {
  1731. string s = printedText;
  1732.  
  1733. if (characterIndex != -1 && characterIndex < s.Length - 6)
  1734. {
  1735. int linkStart;
  1736.  
  1737. // LastIndexOf() fails if the string happens to begin with the expected text
  1738. if (s[characterIndex] == '[' &&
  1739. s[characterIndex + 1] == 'u' &&
  1740. s[characterIndex + 2] == 'r' &&
  1741. s[characterIndex + 3] == 'l' &&
  1742. s[characterIndex + 4] == '=')
  1743. {
  1744. linkStart = characterIndex;
  1745. }
  1746. else linkStart = s.LastIndexOf("[url=", characterIndex);
  1747. if (linkStart == -1) return null;
  1748.  
  1749. linkStart += 5;
  1750. int linkEnd = s.IndexOf("]", linkStart);
  1751. if (linkEnd == -1) return null;
  1752.  
  1753. int urlEnd = s.IndexOf("[/url]", linkEnd);
  1754. if (urlEnd == -1 || characterIndex <= urlEnd)
  1755. return s.Substring(linkStart, linkEnd - linkStart);
  1756. }
  1757. return null;
  1758. }
  1759.  
  1760. /// <summary>
  1761. /// Get the index of the character on the line directly above or below the current index.
  1762. /// </summary>
  1763.  
  1764. public int GetCharacterIndex (int currentIndex, KeyCode key)
  1765. {
  1766. if (isValid)
  1767. {
  1768. string text = processedText;
  1769. if (string.IsNullOrEmpty(text)) return 0;
  1770.  
  1771. int def = defaultFontSize;
  1772. UpdateNGUIText();
  1773.  
  1774. NGUIText.PrintApproximateCharacterPositions(text, mTempVerts, mTempIndices);
  1775.  
  1776. if (mTempVerts.Count > 0)
  1777. {
  1778. ApplyOffset(mTempVerts, 0);
  1779.  
  1780. for (int i = 0, imax = mTempIndices.Count; i < imax; ++i)
  1781. {
  1782. if (mTempIndices[i] == currentIndex)
  1783. {
  1784. // Determine position on the line above or below this character
  1785. Vector2 localPos = mTempVerts[i];
  1786.  
  1787. if (key == KeyCode.UpArrow) localPos.y += def + effectiveSpacingY;
  1788. else if (key == KeyCode.DownArrow) localPos.y -= def + effectiveSpacingY;
  1789. else if (key == KeyCode.Home) localPos.x -= 1000f;
  1790. else if (key == KeyCode.End) localPos.x += 1000f;
  1791.  
  1792. // Find the closest character to this position
  1793. int retVal = NGUIText.GetApproximateCharacterIndex(mTempVerts, mTempIndices, localPos);
  1794. if (retVal == currentIndex) break;
  1795.  
  1796. mTempVerts.Clear();
  1797. mTempIndices.Clear();
  1798. return retVal;
  1799. }
  1800. }
  1801. mTempVerts.Clear();
  1802. mTempIndices.Clear();
  1803. }
  1804.  
  1805. NGUIText.bitmapFont = null;
  1806. NGUIText.dynamicFont = null;
  1807.  
  1808. // If the selection doesn't move, then we're at the top or bottom-most line
  1809. if (key == KeyCode.UpArrow || key == KeyCode.Home) return 0;
  1810. if (key == KeyCode.DownArrow || key == KeyCode.End) return text.Length;
  1811. }
  1812. return currentIndex;
  1813. }
  1814.  
  1815. /// <summary>
  1816. /// Fill the specified geometry buffer with vertices that would highlight the current selection.
  1817. /// </summary>
  1818.  
  1819. public void PrintOverlay (int start, int end, UIGeometry caret, UIGeometry highlight, Color caretColor, Color highlightColor)
  1820. {
  1821. if (caret != null) caret.Clear();
  1822. if (highlight != null) highlight.Clear();
  1823. if (!isValid) return;
  1824.  
  1825. string text = processedText;
  1826. UpdateNGUIText();
  1827.  
  1828. int startingCaretVerts = caret.verts.Count;
  1829. Vector2 center = new Vector2(0.5f, 0.5f);
  1830. float alpha = finalAlpha;
  1831.  
  1832. // If we have a highlight to work with, fill the buffer
  1833. if (highlight != null && start != end)
  1834. {
  1835. int startingVertices = highlight.verts.Count;
  1836. NGUIText.PrintCaretAndSelection(text, start, end, caret.verts, highlight.verts);
  1837.  
  1838. if (highlight.verts.Count > startingVertices)
  1839. {
  1840. ApplyOffset(highlight.verts, startingVertices);
  1841.  
  1842. Color c = new Color(highlightColor.r, highlightColor.g, highlightColor.b, highlightColor.a * alpha);
  1843.  
  1844. for (int i = startingVertices, imax = highlight.verts.Count; i < imax; ++i)
  1845. {
  1846. highlight.uvs.Add(center);
  1847. highlight.cols.Add(c);
  1848. }
  1849. }
  1850. }
  1851. else NGUIText.PrintCaretAndSelection(text, start, end, caret.verts, null);
  1852.  
  1853. // Fill the caret UVs and colors
  1854. ApplyOffset(caret.verts, startingCaretVerts);
  1855. Color cc = new Color(caretColor.r, caretColor.g, caretColor.b, caretColor.a * alpha);
  1856.  
  1857. for (int i = startingCaretVerts, imax = caret.verts.Count; i < imax; ++i)
  1858. {
  1859. caret.uvs.Add(center);
  1860. caret.cols.Add(cc);
  1861. }
  1862.  
  1863. NGUIText.bitmapFont = null;
  1864. NGUIText.dynamicFont = null;
  1865. }
  1866.  
  1867. /// <summary>
  1868. /// Draw the label.
  1869. /// </summary>
  1870.  
  1871. public override void OnFill (List<Vector3> verts, List<Vector2> uvs, List<Color> cols)
  1872. {
  1873. if (!isValid) return;
  1874.  
  1875. int offset = verts.Count;
  1876. Color col = color;
  1877. col.a = finalAlpha;
  1878. if (mFont != null && mFont.premultipliedAlphaShader) col = NGUITools.ApplyPMA(col);
  1879.  
  1880. string text = processedText;
  1881. int start = verts.Count;
  1882.  
  1883. UpdateNGUIText();
  1884.  
  1885. NGUIText.tint = col;
  1886. NGUIText.Print(text, verts, uvs, cols);
  1887. NGUIText.bitmapFont = null;
  1888. NGUIText.dynamicFont = null;
  1889.  
  1890. // Center the content within the label vertically
  1891. Vector2 pos = ApplyOffset(verts, start);
  1892.  
  1893. // Effects don't work with packed fonts
  1894. if (mFont != null && mFont.packedFontShader) return;
  1895.  
  1896. // Apply an effect if one was requested
  1897. if (effectStyle != Effect.None)
  1898. {
  1899. int end = verts.Count;
  1900. pos.x = mEffectDistance.x;
  1901. pos.y = mEffectDistance.y;
  1902.  
  1903. ApplyShadow(verts, uvs, cols, offset, end, pos.x, -pos.y);
  1904.  
  1905. if ((effectStyle == Effect.Outline) || (effectStyle == Effect.Outline8))
  1906. {
  1907. offset = end;
  1908. end = verts.Count;
  1909.  
  1910. ApplyShadow(verts, uvs, cols, offset, end, -pos.x, pos.y);
  1911.  
  1912. offset = end;
  1913. end = verts.Count;
  1914.  
  1915. ApplyShadow(verts, uvs, cols, offset, end, pos.x, pos.y);
  1916.  
  1917. offset = end;
  1918. end = verts.Count;
  1919.  
  1920. ApplyShadow(verts, uvs, cols, offset, end, -pos.x, -pos.y);
  1921.  
  1922. if (effectStyle == Effect.Outline8)
  1923. {
  1924. offset = end;
  1925. end = verts.Count;
  1926.  
  1927. ApplyShadow(verts, uvs, cols, offset, end, -pos.x, 0);
  1928.  
  1929. offset = end;
  1930. end = verts.Count;
  1931.  
  1932. ApplyShadow(verts, uvs, cols, offset, end, pos.x, 0);
  1933.  
  1934. offset = end;
  1935. end = verts.Count;
  1936.  
  1937. ApplyShadow(verts, uvs, cols, offset, end, 0, pos.y);
  1938.  
  1939. offset = end;
  1940. end = verts.Count;
  1941.  
  1942. ApplyShadow(verts, uvs, cols, offset, end, 0, -pos.y);
  1943. }
  1944. }
  1945. }
  1946.  
  1947. if (onPostFill != null)
  1948. onPostFill(this, offset, verts, uvs, cols);
  1949. }
  1950.  
  1951. /// <summary>
  1952. /// Align the vertices, making the label positioned correctly based on the pivot.
  1953. /// Returns the offset that was applied.
  1954. /// </summary>
  1955.  
  1956. public Vector2 ApplyOffset (List<Vector3> verts, int start)
  1957. {
  1958. Vector2 po = pivotOffset;
  1959.  
  1960. float fx = Mathf.Lerp(0f, -mWidth, po.x);
  1961. float fy = Mathf.Lerp(mHeight, 0f, po.y) + Mathf.Lerp((mCalculatedSize.y - mHeight), 0f, po.y);
  1962.  
  1963. fx = Mathf.Round(fx);
  1964. fy = Mathf.Round(fy);
  1965.  
  1966. Vector3 v;
  1967.  
  1968. for (int i = start, imax = verts.Count; i < imax; ++i)
  1969. {
  1970. v = verts[i];
  1971. v.x += fx;
  1972. v.y += fy;
  1973. verts[i] = v;
  1974. }
  1975. return new Vector2(fx, fy);
  1976. }
  1977.  
  1978. /// <summary>
  1979. /// Apply a shadow effect to the buffer.
  1980. /// </summary>
  1981.  
  1982. public void ApplyShadow (List<Vector3> verts, List<Vector2> uvs, List<Color> cols, int start, int end, float x, float y)
  1983. {
  1984. Color c = mEffectColor;
  1985. c.a *= finalAlpha;
  1986. if (bitmapFont != null && bitmapFont.premultipliedAlphaShader) c = NGUITools.ApplyPMA(c);
  1987. Color col = c;
  1988.  
  1989. for (int i = start; i < end; ++i)
  1990. {
  1991. verts.Add(verts[i]);
  1992. uvs.Add(uvs[i]);
  1993. cols.Add(cols[i]);
  1994.  
  1995. var v = verts[i];
  1996. v.x += x;
  1997. v.y += y;
  1998. verts[i] = v;
  1999.  
  2000. Color uc = cols[i];
  2001.  
  2002. if (uc.a == 1f)
  2003. {
  2004. cols[i] = col;
  2005. }
  2006. else
  2007. {
  2008. Color fc = c;
  2009. fc.a = uc.a * c.a;
  2010. cols[i] = fc;
  2011. }
  2012. }
  2013. }
  2014.  
  2015. /// <summary>
  2016. /// Calculate the character index offset necessary in order to print the end of the specified text.
  2017. /// </summary>
  2018.  
  2019. public int CalculateOffsetToFit (string text)
  2020. {
  2021. UpdateNGUIText();
  2022. NGUIText.encoding = false;
  2023. NGUIText.symbolStyle = NGUIText.SymbolStyle.None;
  2024. int offset = NGUIText.CalculateOffsetToFit(text);
  2025. NGUIText.bitmapFont = null;
  2026. NGUIText.dynamicFont = null;
  2027. return offset;
  2028. }
  2029.  
  2030. /// <summary>
  2031. /// Convenience function, in case you wanted to associate progress bar, slider or scroll bar's
  2032. /// OnValueChanged function in inspector with a label.
  2033. /// </summary>
  2034.  
  2035. public void SetCurrentProgress ()
  2036. {
  2037. if (UIProgressBar.current != null)
  2038. text = UIProgressBar.current.value.ToString("F");
  2039. }
  2040.  
  2041. /// <summary>
  2042. /// Convenience function, in case you wanted to associate progress bar, slider or scroll bar's
  2043. /// OnValueChanged function in inspector with a label.
  2044. /// </summary>
  2045.  
  2046. public void SetCurrentPercent ()
  2047. {
  2048. if (UIProgressBar.current != null)
  2049. text = Mathf.RoundToInt(UIProgressBar.current.value * 100f) + "%";
  2050. }
  2051.  
  2052. /// <summary>
  2053. /// Convenience function, in case you wanted to automatically set some label's text
  2054. /// by selecting a value in the UIPopupList.
  2055. /// </summary>
  2056.  
  2057. public void SetCurrentSelection ()
  2058. {
  2059. if (UIPopupList.current != null)
  2060. {
  2061. text = UIPopupList.current.isLocalized ?
  2062. Localization.Get(UIPopupList.current.value) :
  2063. UIPopupList.current.value;
  2064. }
  2065. }
  2066.  
  2067. /// <summary>
  2068. /// Convenience function -- wrap the current text given the label's settings and unlimited height.
  2069. /// </summary>
  2070.  
  2071. public bool Wrap (string text, out string final) { return Wrap(text, out final, 1000000); }
  2072.  
  2073. /// <summary>
  2074. /// Convenience function -- wrap the current text given the label's settings and the given height.
  2075. /// </summary>
  2076.  
  2077. public bool Wrap (string text, out string final, int height)
  2078. {
  2079. UpdateNGUIText();
  2080. NGUIText.rectHeight = height;
  2081. NGUIText.regionHeight = height;
  2082. bool retVal = NGUIText.WrapText(text, out final);
  2083. NGUIText.bitmapFont = null;
  2084. NGUIText.dynamicFont = null;
  2085. return retVal;
  2086. }
  2087.  
  2088. /// <summary>
  2089. /// Update NGUIText.current with all the properties from this label.
  2090. /// </summary>
  2091.  
  2092. public void UpdateNGUIText ()
  2093. {
  2094. Font ttf = trueTypeFont;
  2095. bool isDynamic = (ttf != null);
  2096.  
  2097. UpdateFontStyle();
  2098. NGUIText.fontSize = mFinalFontSize;
  2099. NGUIText.fontStyle = mFontStyle;
  2100. NGUIText.rectWidth = mWidth;
  2101. NGUIText.rectHeight = mHeight;
  2102. NGUIText.regionWidth = Mathf.RoundToInt(mWidth * (mDrawRegion.z - mDrawRegion.x));
  2103. NGUIText.regionHeight = Mathf.RoundToInt(mHeight * (mDrawRegion.w - mDrawRegion.y));
  2104. NGUIText.gradient = mApplyGradient && (mFont == null || !mFont.packedFontShader);
  2105. NGUIText.gradientTop = mGradientTop;
  2106. NGUIText.gradientBottom = mGradientBottom;
  2107. NGUIText.encoding = mEncoding;
  2108. NGUIText.premultiply = mPremultiply;
  2109. NGUIText.symbolStyle = mSymbols;
  2110. NGUIText.maxLines = mMaxLineCount;
  2111. NGUIText.spacingX = effectiveSpacingX;
  2112. NGUIText.spacingY = effectiveSpacingY;
  2113. NGUIText.fontScale = isDynamic ? mScale : ((float)mFontSize / mFont.defaultSize) * mScale;
  2114.  
  2115. if (mFont != null)
  2116. {
  2117. NGUIText.bitmapFont = mFont;
  2118. for (; ; )
  2119. {
  2120. UIFont fnt = NGUIText.bitmapFont.replacement;
  2121. if (fnt == null) break;
  2122. NGUIText.bitmapFont = fnt;
  2123. }
  2124.  
  2125. if (NGUIText.bitmapFont.isDynamic)
  2126. {
  2127. NGUIText.dynamicFont = NGUIText.bitmapFont.dynamicFont;
  2128. NGUIText.bitmapFont = null;
  2129. }
  2130. else NGUIText.dynamicFont = null;
  2131. }
  2132. else
  2133. {
  2134. NGUIText.dynamicFont = ttf;
  2135. NGUIText.bitmapFont = null;
  2136. }
  2137.  
  2138. if (isDynamic && keepCrisp)
  2139. {
  2140. UIRoot rt = root;
  2141. if (rt != null) NGUIText.pixelDensity = (rt != null) ? rt.pixelSizeAdjustment : 1f;
  2142. }
  2143. else NGUIText.pixelDensity = 1f;
  2144.  
  2145. if (mDensity != NGUIText.pixelDensity)
  2146. {
  2147. ProcessText(false, false);
  2148. NGUIText.rectWidth = mWidth;
  2149. NGUIText.rectHeight = mHeight;
  2150. NGUIText.regionWidth = Mathf.RoundToInt(mWidth * (mDrawRegion.z - mDrawRegion.x));
  2151. NGUIText.regionHeight = Mathf.RoundToInt(mHeight * (mDrawRegion.w - mDrawRegion.y));
  2152. }
  2153.  
  2154. if (alignment == Alignment.Automatic)
  2155. {
  2156. Pivot p = pivot;
  2157.  
  2158. if (p == Pivot.Left || p == Pivot.TopLeft || p == Pivot.BottomLeft)
  2159. {
  2160. NGUIText.alignment = Alignment.Left;
  2161. }
  2162. else if (p == Pivot.Right || p == Pivot.TopRight || p == Pivot.BottomRight)
  2163. {
  2164. NGUIText.alignment = Alignment.Right;
  2165. }
  2166. else NGUIText.alignment = Alignment.Center;
  2167. }
  2168. else NGUIText.alignment = alignment;
  2169.  
  2170. NGUIText.Update();
  2171. }
  2172.  
  2173. void OnApplicationPause (bool paused)
  2174. {
  2175. if (!paused && mTrueTypeFont != null) Invalidate(false);
  2176. }
  2177.  
  2178.  
  2179.  
  2180. #region 预定义几种常用字体样式
  2181. public enum FontStyleEnum
  2182. {
  2183. None,
  2184. Normal,
  2185. Tab1,
  2186. Tab2,
  2187. Button1,
  2188. Button2,
  2189. Description,
  2190. AttributeNumber,
  2191. Quality1,
  2192. Quality2,
  2193. Quality3,
  2194. Quality4,
  2195. Quality5,
  2196. Red,
  2197. Green,
  2198. LightBlue,
  2199. Yellow,
  2200. GradientYellow,
  2201. Gray
  2202. }
  2203. [HideInInspector]
  2204. [SerializeField]
  2205. FontStyleEnum mFontStyleEnum = FontStyleEnum.None;
  2206.  
  2207. static List<CommonFontStyle> fontStyleList = null;
  2208. class CommonFontStyle
  2209. {
  2210. public int fontSize = 24;
  2211. public FontStyle fontStyle = FontStyle.Normal;
  2212. public bool applyGradient = false; //是否应用渐变
  2213. public Color gradientTop = Color.white;
  2214. public Color gradientBottom = Color.yellow;
  2215. public Color color = Color.white;
  2216. public UILabel.Effect effectStyle = UILabel.Effect.None;
  2217. public Color effectColor = Color.black;
  2218. public string description = "描述信息";
  2219. }
  2220.  
  2221. public static void LoadFontStyleConfig()
  2222. {
  2223. #if UNITY_EDITOR
  2224. string fileName = Directory.GetCurrentDirectory() + @"\Assets\NGUI\fontstyle.txt";
  2225. if (!File.Exists(fileName))
  2226. {
  2227. Debug.Log("Not found fontstyle.txt");
  2228. return;
  2229. }
  2230.  
  2231. if (null == fontStyleList)
  2232. fontStyleList = new List<CommonFontStyle>();
  2233. fontStyleList.Clear();
  2234. fontStyleList.Add(null);//0号元素为 None
  2235.  
  2236. string text = File.ReadAllText(fileName);
  2237. JsonReader reader = new JsonReader(text);
  2238. JsonData arr = JsonMapper.ToObject(reader);
  2239. if (!arr.IsArray)
  2240. return;
  2241.  
  2242. double r, g, b, a;
  2243. for (int i = 0; i < arr.Count; i++)
  2244. {
  2245. JsonData data = arr[i];
  2246.  
  2247. CommonFontStyle cfs = new CommonFontStyle();
  2248. cfs.fontSize = (int)data["fontSize"];
  2249. cfs.fontStyle = (FontStyle)Enum.Parse(typeof(FontStyle), (string)data["fontStyle"]);
  2250. cfs.applyGradient = (bool)data["applyGradient"];
  2251.  
  2252. JsonParseColor(data, "gradientTop", out r, out g, out b, out a);
  2253. cfs.gradientTop = new Color((float)r, (float)g, (float)b, (float)a);
  2254.  
  2255. JsonParseColor(data, "gradientBottom", out r, out g, out b, out a);
  2256. cfs.gradientBottom = new Color((float)r, (float)g, (float)b, (float)a);
  2257.  
  2258. JsonParseColor(data, "color", out r, out g, out b, out a);
  2259. cfs.color = new Color((float)r, (float)g, (float)b, (float)a);
  2260. cfs.effectStyle = (UILabel.Effect)Enum.Parse(typeof(UILabel.Effect), (string)data["effectStyle"]);
  2261.  
  2262. JsonParseColor(data, "effectColor", out r, out g, out b, out a);
  2263. cfs.effectColor = new Color((float)r, (float)g, (float)b, (float)a);
  2264.  
  2265. cfs.description = (string)data["description"];
  2266. fontStyleList.Add(cfs);
  2267. }
  2268.  
  2269. Debug.Log("Load fontstyle.txt completed!");
  2270. #endif
  2271. }
  2272.  
  2273. #if UNITY_EDITOR
  2274. static void JsonParseColor(JsonData data, string key, out double r, out double g, out double b, out double a)
  2275. {
  2276. r = (double)data[key]["r"];
  2277. g = (double)data[key]["g"];
  2278. b = (double)data[key]["b"];
  2279. a = (double)data[key]["a"];
  2280. }
  2281. #endif
  2282.  
  2283. void SetLabelFontStyle(FontStyleEnum fontStyleEnum)
  2284. {
  2285. if (fontStyleEnum == FontStyleEnum.None)
  2286. return;
  2287.  
  2288. if (null == fontStyleList)
  2289. LoadFontStyleConfig();
  2290.  
  2291. if (null == fontStyleList)
  2292. return;
  2293.  
  2294. int index = (int)fontStyleEnum;
  2295. CommonFontStyle style = null;
  2296.  
  2297. if (index >= fontStyleList.Count)
  2298. return;
  2299.  
  2300. style = fontStyleList[index];
  2301. if (null == style)
  2302. return;
  2303. this.fontStyle = style.fontStyle;
  2304. this.fontSize = style.fontSize;
  2305. this.applyGradient = style.applyGradient;
  2306. this.gradientTop = style.gradientTop;
  2307. this.gradientBottom = style.gradientBottom;
  2308. this.color = style.color;
  2309. this.effectStyle = style.effectStyle;
  2310. this.effectColor = style.effectColor;
  2311. this.MarkAsChanged();
  2312. }
  2313.  
  2314. public void UpdateFontStyle()
  2315. {
  2316. SetLabelFontStyle(mFontStyleEnum);
  2317. }
  2318.  
  2319. public string GetFontStyleDescription()
  2320. {
  2321. if (mFontStyleEnum == FontStyleEnum.None)
  2322. return string.Empty;
  2323. int index = (int)mFontStyleEnum;
  2324. if (null == fontStyleList || index >= fontStyleList.Count)
  2325. return string.Empty;
  2326. CommonFontStyle style = fontStyleList[index];
  2327. if (null == style)
  2328. return string.Empty;
  2329. return style.description;
  2330. }
  2331. #endregion
  2332. }


二、修改UILabelInspector.cs

  1. //-------------------------------------------------
  2. // NGUI: Next-Gen UI kit
  3. // Copyright © 2011-2017 Tasharen Entertainment Inc
  4. //-------------------------------------------------
  5.  
  6. #if !UNITY_FLASH
  7. #define DYNAMIC_FONT
  8. #endif
  9.  
  10. using UnityEngine;
  11. using UnityEditor;
  12.  
  13. /// <summary>
  14. /// Inspector class used to edit UILabels.
  15. /// </summary>
  16.  
  17. [CanEditMultipleObjects]
  18. [CustomEditor(typeof(UILabel), true)]
  19. public class UILabelInspector : UIWidgetInspector
  20. {
  21. public enum FontType
  22. {
  23. NGUI,
  24. Unity,
  25. }
  26.  
  27. UILabel mLabel;
  28. FontType mFontType;
  29.  
  30. protected override void OnEnable ()
  31. {
  32. base.OnEnable();
  33. SerializedProperty bit = serializedObject.FindProperty("mFont");
  34. mFontType = (bit != null && bit.objectReferenceValue != null) ? FontType.NGUI : FontType.Unity;
  35. }
  36.  
  37. void OnNGUIFont (Object obj)
  38. {
  39. serializedObject.Update();
  40. SerializedProperty sp = serializedObject.FindProperty("mFont");
  41. sp.objectReferenceValue = obj;
  42.  
  43. sp = serializedObject.FindProperty("mTrueTypeFont");
  44. sp.objectReferenceValue = null;
  45. serializedObject.ApplyModifiedProperties();
  46. NGUISettings.ambigiousFont = obj;
  47. }
  48.  
  49. void OnUnityFont (Object obj)
  50. {
  51. serializedObject.Update();
  52. SerializedProperty sp = serializedObject.FindProperty("mTrueTypeFont");
  53. sp.objectReferenceValue = obj;
  54.  
  55. sp = serializedObject.FindProperty("mFont");
  56. sp.objectReferenceValue = null;
  57.  
  58. serializedObject.ApplyModifiedProperties();
  59. NGUISettings.ambigiousFont = obj;
  60. }
  61.  
  62. /// <summary>
  63. /// Draw the label's properties.
  64. /// </summary>
  65.  
  66. protected override bool ShouldDrawProperties ()
  67. {
  68. mLabel = mWidget as UILabel;
  69.  
  70. GUILayout.BeginHorizontal();
  71.  
  72. #if DYNAMIC_FONT
  73. mFontType = (FontType)EditorGUILayout.EnumPopup(mFontType, "DropDown", GUILayout.Width(74f));
  74. if (NGUIEditorTools.DrawPrefixButton("Font", GUILayout.Width(64f)))
  75. #else
  76. mFontType = FontType.NGUI;
  77. if (NGUIEditorTools.DrawPrefixButton("Font", GUILayout.Width(74f)))
  78. #endif
  79. {
  80. if (mFontType == FontType.NGUI)
  81. {
  82. ComponentSelector.Show<UIFont>(OnNGUIFont);
  83. }
  84. else
  85. {
  86. ComponentSelector.Show<Font>(OnUnityFont, new string[] { ".ttf", ".otf" });
  87. }
  88. }
  89.  
  90. bool isValid = false;
  91. SerializedProperty fnt = null;
  92. SerializedProperty ttf = null;
  93.  
  94. if (mFontType == FontType.NGUI)
  95. {
  96. GUI.changed = false;
  97. fnt = NGUIEditorTools.DrawProperty("", serializedObject, "mFont", GUILayout.MinWidth(40f));
  98.  
  99. if (fnt.objectReferenceValue != null)
  100. {
  101. if (GUI.changed) serializedObject.FindProperty("mTrueTypeFont").objectReferenceValue = null;
  102. NGUISettings.ambigiousFont = fnt.objectReferenceValue;
  103. isValid = true;
  104. }
  105. }
  106. else
  107. {
  108. GUI.changed = false;
  109. ttf = NGUIEditorTools.DrawProperty("", serializedObject, "mTrueTypeFont", GUILayout.MinWidth(40f));
  110.  
  111. if (ttf.objectReferenceValue != null)
  112. {
  113. if (GUI.changed) serializedObject.FindProperty("mFont").objectReferenceValue = null;
  114. NGUISettings.ambigiousFont = ttf.objectReferenceValue;
  115. isValid = true;
  116. }
  117. }
  118.  
  119. GUILayout.EndHorizontal();
  120.  
  121. //if (mFontType == FontType.Unity)
  122. //{
  123. // EditorGUILayout.HelpBox("Dynamic fonts suffer from issues in Unity itself where your characters may disappear, get garbled, or just not show at times. Use this feature at your own risk.\n\n" +
  124. // "When you do run into such issues, please submit a Bug Report to Unity via Help -> Report a Bug (as this is will be a Unity bug, not an NGUI one).", MessageType.Warning);
  125. //}
  126.  
  127. NGUIEditorTools.DrawProperty("Material", serializedObject, "mMat");
  128.  
  129. #region 新增常用字体样式
  130. GUI.changed = false;
  131. Rect pRect = EditorGUILayout.GetControlRect(GUILayout.Height(16));
  132. GUIContent fs_label = new GUIContent("Font Style");
  133. SerializedProperty fs_sp = serializedObject.FindProperty("mFontStyleEnum");
  134. UILabel.FontStyleEnum mFontStyleIndex = (UILabel.FontStyleEnum)EditorGUI.EnumPopup(pRect, fs_label, (UILabel.FontStyleEnum)fs_sp.enumValueIndex);
  135. if (GUI.changed) {
  136. fs_sp.enumValueIndex = (int)mFontStyleIndex;
  137. }
  138. if (mFontStyleIndex != UILabel.FontStyleEnum.None && string.Empty != mLabel.GetFontStyleDescription())
  139. {
  140. EditorGUILayout.HelpBox(mLabel.GetFontStyleDescription(), MessageType.Info);
  141. }
  142. #endregion
  143.  
  144. EditorGUI.BeginDisabledGroup(!isValid);
  145. {
  146. UIFont uiFont = (fnt != null) ? fnt.objectReferenceValue as UIFont : null;
  147. Font dynFont = (ttf != null) ? ttf.objectReferenceValue as Font : null;
  148.  
  149. if (uiFont != null && uiFont.isDynamic)
  150. {
  151. dynFont = uiFont.dynamicFont;
  152. uiFont = null;
  153. }
  154.  
  155. if (dynFont != null)
  156. {
  157. GUILayout.BeginHorizontal();
  158. {
  159. EditorGUI.BeginDisabledGroup((ttf != null) ? ttf.hasMultipleDifferentValues : fnt.hasMultipleDifferentValues);
  160. SerializedProperty prop = NGUIEditorTools.DrawProperty("Font Size", serializedObject, "mFontSize", GUILayout.Width(142f));
  161. NGUISettings.fontSize = prop.intValue;
  162. prop = NGUIEditorTools.DrawProperty("", serializedObject, "mFontStyle", GUILayout.MinWidth(40f));
  163. NGUISettings.fontStyle = (FontStyle)prop.intValue;
  164. NGUIEditorTools.DrawPadding();
  165. EditorGUI.EndDisabledGroup();
  166. }
  167. GUILayout.EndHorizontal();
  168. }
  169. else if (uiFont != null)
  170. {
  171. GUILayout.BeginHorizontal();
  172. SerializedProperty prop = NGUIEditorTools.DrawProperty("Font Size", serializedObject, "mFontSize", GUILayout.Width(142f));
  173.  
  174. EditorGUI.BeginDisabledGroup(true);
  175. if (!serializedObject.isEditingMultipleObjects)
  176. {
  177. if (mLabel.overflowMethod == UILabel.Overflow.ShrinkContent)
  178. GUILayout.Label(" Actual: " + mLabel.finalFontSize + "/" + mLabel.defaultFontSize);
  179. else GUILayout.Label(" Default: " + mLabel.defaultFontSize);
  180. }
  181. EditorGUI.EndDisabledGroup();
  182.  
  183. NGUISettings.fontSize = prop.intValue;
  184. GUILayout.EndHorizontal();
  185. }
  186.  
  187. bool ww = GUI.skin.textField.wordWrap;
  188. GUI.skin.textField.wordWrap = true;
  189. SerializedProperty sp = serializedObject.FindProperty("mText");
  190.  
  191. if (sp.hasMultipleDifferentValues)
  192. {
  193. NGUIEditorTools.DrawProperty("", sp, GUILayout.Height(128f));
  194. }
  195. else
  196. {
  197. GUIStyle style = new GUIStyle(EditorStyles.textField);
  198. style.wordWrap = true;
  199.  
  200. float height = style.CalcHeight(new GUIContent(sp.stringValue), Screen.width - 100f);
  201. bool offset = true;
  202.  
  203. if (height > 90f)
  204. {
  205. offset = false;
  206. height = style.CalcHeight(new GUIContent(sp.stringValue), Screen.width - 20f);
  207. }
  208. else
  209. {
  210. GUILayout.BeginHorizontal();
  211. GUILayout.BeginVertical(GUILayout.Width(76f));
  212. GUILayout.Space(3f);
  213. GUILayout.Label("Text");
  214. GUILayout.EndVertical();
  215. GUILayout.BeginVertical();
  216. }
  217. Rect rect = EditorGUILayout.GetControlRect(GUILayout.Height(height));
  218.  
  219. GUI.changed = false;
  220. string text = EditorGUI.TextArea(rect, sp.stringValue, style);
  221. if (GUI.changed) sp.stringValue = text;
  222.  
  223. if (offset)
  224. {
  225. GUILayout.EndVertical();
  226. GUILayout.EndHorizontal();
  227. }
  228. }
  229.  
  230. GUI.skin.textField.wordWrap = ww;
  231.  
  232. NGUIEditorTools.DrawPaddedProperty("Modifier", serializedObject, "mModifier");
  233.  
  234. SerializedProperty ov = NGUIEditorTools.DrawPaddedProperty("Overflow", serializedObject, "mOverflow");
  235. NGUISettings.overflowStyle = (UILabel.Overflow)ov.intValue;
  236. if (NGUISettings.overflowStyle == UILabel.Overflow.ClampContent)
  237. NGUIEditorTools.DrawProperty("Use Ellipsis", serializedObject, "mOverflowEllipsis", GUILayout.Width(110f));
  238.  
  239. if (NGUISettings.overflowStyle == UILabel.Overflow.ResizeFreely)
  240. {
  241. GUILayout.BeginHorizontal();
  242. SerializedProperty s = NGUIEditorTools.DrawPaddedProperty("Max Width", serializedObject, "mOverflowWidth");
  243. if (s != null && s.intValue < 1) GUILayout.Label("unlimited");
  244. GUILayout.EndHorizontal();
  245. }
  246.  
  247. NGUIEditorTools.DrawPaddedProperty("Alignment", serializedObject, "mAlignment");
  248.  
  249. if (dynFont != null)
  250. NGUIEditorTools.DrawPaddedProperty("Keep crisp", serializedObject, "keepCrispWhenShrunk");
  251.  
  252. EditorGUI.BeginDisabledGroup(mLabel.bitmapFont != null && mLabel.bitmapFont.packedFontShader);
  253. GUILayout.BeginHorizontal();
  254. SerializedProperty gr = NGUIEditorTools.DrawProperty("Gradient", serializedObject, "mApplyGradient",
  255. GUILayout.Width(95f));
  256.  
  257. EditorGUI.BeginDisabledGroup(!gr.hasMultipleDifferentValues && !gr.boolValue);
  258. {
  259. NGUIEditorTools.SetLabelWidth(30f);
  260. NGUIEditorTools.DrawProperty("Top", serializedObject, "mGradientTop", GUILayout.MinWidth(40f));
  261. GUILayout.EndHorizontal();
  262. GUILayout.BeginHorizontal();
  263. NGUIEditorTools.SetLabelWidth(50f);
  264. GUILayout.Space(79f);
  265.  
  266. NGUIEditorTools.DrawProperty("Bottom", serializedObject, "mGradientBottom", GUILayout.MinWidth(40f));
  267. NGUIEditorTools.SetLabelWidth(80f);
  268. }
  269. EditorGUI.EndDisabledGroup();
  270. GUILayout.EndHorizontal();
  271.  
  272. GUILayout.BeginHorizontal();
  273. GUILayout.Label("Effect", GUILayout.Width(76f));
  274. sp = NGUIEditorTools.DrawProperty("", serializedObject, "mEffectStyle", GUILayout.MinWidth(16f));
  275.  
  276. EditorGUI.BeginDisabledGroup(!sp.hasMultipleDifferentValues && !sp.boolValue);
  277. {
  278. NGUIEditorTools.DrawProperty("", serializedObject, "mEffectColor", GUILayout.MinWidth(10f));
  279. GUILayout.EndHorizontal();
  280.  
  281. GUILayout.BeginHorizontal();
  282. {
  283. GUILayout.Label(" ", GUILayout.Width(56f));
  284. NGUIEditorTools.SetLabelWidth(20f);
  285. NGUIEditorTools.DrawProperty("X", serializedObject, "mEffectDistance.x", GUILayout.MinWidth(40f));
  286. NGUIEditorTools.DrawProperty("Y", serializedObject, "mEffectDistance.y", GUILayout.MinWidth(40f));
  287. NGUIEditorTools.DrawPadding();
  288. NGUIEditorTools.SetLabelWidth(80f);
  289. }
  290. }
  291. EditorGUI.EndDisabledGroup();
  292. GUILayout.EndHorizontal();
  293. EditorGUI.EndDisabledGroup();
  294.  
  295. sp = NGUIEditorTools.DrawProperty("Float spacing", serializedObject, "mUseFloatSpacing", GUILayout.Width(100f));
  296.  
  297. if (!sp.boolValue)
  298. {
  299. GUILayout.BeginHorizontal();
  300. GUILayout.Label("Spacing", GUILayout.Width(56f));
  301. NGUIEditorTools.SetLabelWidth(20f);
  302. NGUIEditorTools.DrawProperty("X", serializedObject, "mSpacingX", GUILayout.MinWidth(40f));
  303. NGUIEditorTools.DrawProperty("Y", serializedObject, "mSpacingY", GUILayout.MinWidth(40f));
  304. NGUIEditorTools.DrawPadding();
  305. NGUIEditorTools.SetLabelWidth(80f);
  306. GUILayout.EndHorizontal();
  307. }
  308. else
  309. {
  310. GUILayout.BeginHorizontal();
  311. GUILayout.Label("Spacing", GUILayout.Width(56f));
  312. NGUIEditorTools.SetLabelWidth(20f);
  313. NGUIEditorTools.DrawProperty("X", serializedObject, "mFloatSpacingX", GUILayout.MinWidth(40f));
  314. NGUIEditorTools.DrawProperty("Y", serializedObject, "mFloatSpacingY", GUILayout.MinWidth(40f));
  315. NGUIEditorTools.DrawPadding();
  316. NGUIEditorTools.SetLabelWidth(80f);
  317. GUILayout.EndHorizontal();
  318. }
  319. NGUIEditorTools.DrawProperty("Max Lines", serializedObject, "mMaxLineCount", GUILayout.Width(110f));
  320.  
  321. GUILayout.BeginHorizontal();
  322. sp = NGUIEditorTools.DrawProperty("BBCode", serializedObject, "mEncoding", GUILayout.Width(100f));
  323. EditorGUI.BeginDisabledGroup(!sp.boolValue || mLabel.bitmapFont == null || !mLabel.bitmapFont.hasSymbols);
  324. NGUIEditorTools.SetLabelWidth(60f);
  325. NGUIEditorTools.DrawPaddedProperty("Symbols", serializedObject, "mSymbols");
  326. NGUIEditorTools.SetLabelWidth(80f);
  327. EditorGUI.EndDisabledGroup();
  328. GUILayout.EndHorizontal();
  329. }
  330. EditorGUI.EndDisabledGroup();
  331. return isValid;
  332. }
  333. }


三、添加导出字体设置信息菜单

  1. using UnityEngine;
  2. using UnityEditor;
  3. using System.Collections;
  4. using System.Collections.Generic;
  5. using System;
  6. using System.IO;
  7.  
  8. static public class UIToolsEditor
  9. {
  10. #region UILabel 导出字体样式设置
  11. [MenuItem("NGUI/Label Setting/Output Setting")]
  12. static void LabelOutputSetting() {
  13. GameObject[] gameObjects = Selection.gameObjects;
  14. if (null == gameObjects)
  15. return;
  16.  
  17. StringBuilder sb = new StringBuilder();
  18. for (int i = 0; i < gameObjects.Length; i++) {
  19. GameObject go = gameObjects[i];
  20. UILabel label = go.GetComponent<UILabel>();
  21. if (null == label)
  22. continue;
  23.  
  24. CommonFontStyleJson cfs = new CommonFontStyleJson();
  25. cfs.fontSize = label.fontSize;
  26. cfs.applyGradient = label.applyGradient;
  27. cfs.gradientTop = new CommonFontStyleColor(label.gradientTop.r, label.gradientTop.g, label.gradientTop.b, label.gradientTop.a);
  28. cfs.gradientBottom = new CommonFontStyleColor(label.gradientBottom.r, label.gradientBottom.g, label.gradientBottom.b, label.gradientBottom.a);
  29. cfs.color = new CommonFontStyleColor(label.color.r, label.color.g, label.color.b, label.color.a);
  30. cfs.effectStyle = label.effectStyle.ToString();
  31. cfs.effectColor = new CommonFontStyleColor(label.effectColor.r, label.effectColor.g, label.effectColor.b, label.effectColor.a);
  32. Debug.Log(LitJson.JsonMapper.ToJson(cfs, true));
  33.  
  34. sb.AppendLine(LitJson.JsonMapper.ToJson(cfs, true));
  35. }
  36.  
  37. if (sb.Length > 0)
  38. {
  39. string font_str = sb.ToString();
  40. byte[] font_bytes = Encoding.UTF8.GetBytes(font_str.ToCharArray());
  41.  
  42. string font_file = Application.dataPath + "/NGUI/fontstyle_tmp.txt";
  43. FileStream fs = File.OpenWrite(font_file);
  44. fs.Write(font_bytes, 0, font_bytes.Length);
  45. fs.Flush();
  46. AssetDatabase.Refresh();
  47. Debug.Log("Output: "+font_file);
  48. }
  49. }
  50. [MenuItem("NGUI/Label Setting/Reload fontstyle.txt")]
  51. static void ReloadFontStyleConfig() {
  52. UILabel.LoadFontStyleConfig();
  53. }
  54.  
  55. public class CommonFontStyleJson {
  56. public int fontSize;
  57. public bool applyGradient;
  58. public CommonFontStyleColor gradientTop;
  59. public CommonFontStyleColor gradientBottom;
  60. public CommonFontStyleColor color;
  61. public string effectStyle;
  62. public CommonFontStyleColor effectColor;
  63. public string description = "描述信息";
  64. }
  65.  
  66. public class CommonFontStyleColor {
  67. //LitJson不支持float
  68. public double r;
  69. public double g;
  70. public double b;
  71. public double a;
  72.  
  73. public CommonFontStyleColor(float r, float g, float b, float a) {
  74. this.r = r;
  75. this.g = g;
  76. this.b = b;
  77. this.a = a;
  78. }
  79. }
  80. #endregion
  81. }

效果截图

1111.png

33333.png

222222.png

标签: NGUI

Powered by emlog  蜀ICP备18021003号-1   sitemap

川公网安备 51019002001593号