using System; using System.Collections.Generic; using UnityEngine; namespace UIWidgets.ui { public class LineBreaker { public class LineInfo { public int start; public double width; } private StyledRuns _runs; public Vector2d[] _characterPositions; public double[] _characterWidth; private string _text; private double _width; private int _lineStart; private int _wordStart; private int _spaceCount = 0; private int tabCount = 4; private double _lineLength; private List _lines; public void setup(string text, StyledRuns runs, double width, Vector2d[] characterPositions, double[] characterWidth) { _text = text; _runs = runs; _characterPositions = characterPositions; _characterWidth = characterWidth; _width = width; } public List getLines() { return _lines; } public void doBreak(int blockStart, int blockEnd) { _lines = new List(); _lineStart = blockStart; _wordStart = blockStart; _spaceCount = 0; double offsetX = 0.0; var runIterator = _runs.iterator(); for (var charIndex = blockStart; charIndex < blockEnd; charIndex++) { runIterator.nextTo(charIndex); var run = runIterator.run; var font = FontManager.instance.getOrCreate(run.style.fontFamily).font; var style = run.style; var charInfo = new CharacterInfo(); if (_text[charIndex] == '\t') { _spaceCount++; font.GetCharacterInfo(' ', out charInfo, style.UnityFontSize, style.UnityFontStyle); double tabSize = charInfo.advance * tabCount; var newX = Math.Floor(((offsetX / tabSize) + 1) * tabSize); if (newX > _width && _lineStart != charIndex) { _characterWidth[charIndex] = tabSize; makeLine(charIndex, charIndex); } else { _characterWidth[charIndex] = newX - offsetX; _characterPositions[charIndex].x = offsetX; } offsetX = _characterPositions[charIndex].x + _characterWidth[charIndex]; } else if (_text[charIndex] == ' ') { font.GetCharacterInfo(_text[charIndex], out charInfo, style.UnityFontSize, run.style.UnityFontStyle); _spaceCount++; _characterPositions[charIndex].x = offsetX; _characterWidth[charIndex] = charInfo.advance; offsetX = _characterPositions[charIndex].x + _characterWidth[charIndex]; // todo no wrap in space ? } else { font.GetCharacterInfo(_text[charIndex], out charInfo, style.UnityFontSize, run.style.UnityFontStyle); if (_spaceCount > 0 || blockStart == charIndex) { _wordStart = charIndex; } _characterPositions[charIndex].x = offsetX; _characterWidth[charIndex] = charInfo.advance; if (offsetX + charInfo.advance > _width && _lineStart != charIndex) { if (_lineStart == _wordStart) { makeLine(charIndex, charIndex); _wordStart = charIndex; } else { makeLine(_wordStart, charIndex); } } offsetX = _characterPositions[charIndex].x + _characterWidth[charIndex]; _spaceCount = 0; } } makeLine(blockEnd, blockEnd); } private void makeLine(int end, int last) { Debug.Assert(_lineStart < end); Debug.Assert(end <= last); _lines.Add(new LineInfo() { start = _lineStart, width = _characterPositions[end - 1].x + _characterWidth[end - 1], }); _lineStart = end; if (end >= _characterPositions.Length) { return; } var offset = new Vector2d(-_characterPositions[end].x, 0); _characterPositions[end].x = 0; if (end < last) { Paragraph.offsetCharacters(offset, _characterPositions, end + 1, last + 1); } } } }