using System; using System.Collections.Generic; using System.Linq; using UnityEngine; namespace Unity.UIWidgets.ui { public class TabStops { int _tabWidth = int.MaxValue; Font _font; int _fontSize; const int kTabSpaceCount = 4; List _stops = new List(); public void set(List stops, int tabWidth) { this._stops.Clear(); if (stops != null) { this._stops.AddRange(stops); } this._tabWidth = tabWidth; } public void setFont(Font font, int size) { if (this._font != font || this._fontSize != size) { this._tabWidth = int.MaxValue; } this._font = font; this._fontSize = size; } public float nextTab(float widthSoFar) { for (int i = 0; i < this._stops.Count; i++) { if (this._stops[i] > widthSoFar) { return this._stops[i]; } } if (this._tabWidth == int.MaxValue) { this._font.RequestCharactersInTexture(" ", this._fontSize); CharacterInfo characterInfo; this._font.GetCharacterInfo(' ', out characterInfo, this._fontSize); this._tabWidth = characterInfo.advance * kTabSpaceCount; } if (this._tabWidth == 0) { return widthSoFar; } return (Mathf.Floor(widthSoFar / this._tabWidth + 1) * this._tabWidth); } } public class Candidate { public int offset; public int pre; public float preBreak; public float penalty; public float postBreak; public int preSpaceCount; public int postSpaceCount; } public class LineBreaker { const float ScoreInfty = float.MaxValue; const float ScoreDesperate = 1e10f; string _textBuf; int _textOffset; int _textLength; List _charWidths = new List(); List _breaks = new List(); List _widths = new List(); WordBreaker _wordBreaker = new WordBreaker(); float _width = 0.0f; float _preBreak; float _lineWidth; int _lastBreak; int _bestBreak; float _bestScore; int _spaceCount; TabStops _tabStops; int mFirstTabIndex; List _candidates = new List(); public int computeBreaks() { int nCand = this._candidates.Count; if (nCand > 0 && (nCand == 1 || this._lastBreak != nCand - 1)) { var cand = this._candidates[this._candidates.Count - 1]; this._pushBreak(cand.offset, (cand.postBreak - this._preBreak)); } return this._breaks.Count; } public List getBreaks() { return this._breaks; } public void resize(int size) { if (this._charWidths.Count < size) { this._charWidths.AddRange(Enumerable.Repeat(0.0f, size - this._charWidths.Count)); } } public void setText(string text, int textOffset, int textLength) { this._textBuf = text; this._textOffset = textOffset; this._textLength = textLength; this._wordBreaker.setText(this._textBuf, textOffset, textLength); this._wordBreaker.next(); this._candidates.Clear(); Candidate can = new Candidate { offset = 0, postBreak = 0, preBreak = 0, postSpaceCount = 0, preSpaceCount = 0, pre = 0 }; this._candidates.Add(can); this._lastBreak = 0; this._bestBreak = 0; this._bestScore = ScoreInfty; this._preBreak = 0; this.mFirstTabIndex = int.MaxValue; this._spaceCount = 0; } public void setLineWidth(float lineWidth) { this._lineWidth = lineWidth; } public float addStyleRun(TextStyle style, int start, int end) { float width = 0.0f; if (style != null) { width = Layout.measureText(this._width - this._preBreak, this._textBuf, start + this._textOffset, end - start, style, this._charWidths, start, this._tabStops); } var font = FontManager.instance.getOrCreate(style.fontFamily).font; int current = this._wordBreaker.current(); int afterWord = start; int lastBreak = start; float lastBreakWidth = this._width; float postBreak = this._width; int postSpaceCount = this._spaceCount; for (int i = start; i < end; i++) { char c = this._textBuf[i + this._textOffset]; if (c == '\t') { this._width = this._preBreak + this._tabStops.nextTab((this._width - this._preBreak)); if (this.mFirstTabIndex == int.MaxValue) { this.mFirstTabIndex = i; } } else { if (LayoutUtils.isWordSpace(c)) { this._spaceCount += 1; } this._width += this._charWidths[i]; if (!LayoutUtils.isLineEndSpace(c)) { postBreak = this._width; postSpaceCount = this._spaceCount; afterWord = i + 1; } } if (i + 1 == current) { int wordStart = this._wordBreaker.wordStart(); int wordEnd = this._wordBreaker.wordEnd(); if (style != null || current == end || this._charWidths[current] > 0) { this._addWordBreak(current, this._width, postBreak, this._spaceCount, postSpaceCount, 0); } lastBreak = current; lastBreakWidth = this._width; current = this._wordBreaker.next(); } } return width; } public void finish() { this._wordBreaker.finish(); this._width = 0; this._candidates.Clear(); this._widths.Clear(); this._breaks.Clear(); this._textBuf = null; } public List getWidths() { return this._widths; } public void setTabStops(TabStops tabStops) { this._tabStops = tabStops; } void _addWordBreak(int offset, float preBreak, float postBreak, int preSpaceCount, int postSpaceCount, float penalty) { Candidate cand = new Candidate(); float width = this._candidates[this._candidates.Count - 1].preBreak; if (postBreak - width > this._lineWidth) { int i = this._candidates[this._candidates.Count - 1].offset; width += this._charWidths[i++]; for (; i < offset; i++) { float w = this._charWidths[i]; if (w > 0) { cand.offset = i; cand.preBreak = width; cand.postBreak = width; cand.preSpaceCount = postSpaceCount; cand.preSpaceCount = postSpaceCount; cand.penalty = ScoreDesperate; this._addCandidate(cand); width += w; } } } cand.offset = offset; cand.preBreak = preBreak; cand.postBreak = postBreak; cand.penalty = penalty; cand.preSpaceCount = preSpaceCount; cand.preSpaceCount = postSpaceCount; this._addCandidate(cand); } void _addCandidate(Candidate cand) { int candIndex = this._candidates.Count; this._candidates.Add(cand); if (cand.postBreak - this._preBreak > this._lineWidth) { this._pushGreedyBreak(); } if (cand.penalty <= this._bestScore) { this._bestBreak = candIndex; this._bestScore = cand.penalty; } } void _pushGreedyBreak() { var bestCandidate = this._candidates[this._bestBreak]; this._pushBreak(bestCandidate.offset, (bestCandidate.postBreak - this._preBreak)); this._bestScore = ScoreInfty; this._lastBreak = this._bestBreak; this._preBreak = bestCandidate.preBreak; } void _pushBreak(int offset, float width) { this._breaks.Add(offset); this._widths.Add(width); } } }