2016-06-11 15:29:45 +10:00
|
|
|
|
// ClassicalSharp copyright 2014-2016 UnknownShadow200 | Licensed under MIT
|
|
|
|
|
using System;
|
|
|
|
|
|
|
|
|
|
namespace ClassicalSharp {
|
|
|
|
|
|
|
|
|
|
public sealed class WrappableStringBuffer : StringBuffer {
|
|
|
|
|
|
|
|
|
|
char[] wrap;
|
|
|
|
|
|
|
|
|
|
public WrappableStringBuffer( int capacity ) : base( capacity ) {
|
|
|
|
|
wrap = new char[capacity];
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public void WordWrap( IDrawer2D drawer, ref string[] lines, ref int[] lineLens,
|
|
|
|
|
int lineSize, int totalChars ) {
|
|
|
|
|
int len = Length;
|
|
|
|
|
for( int i = 0; i < lines.Length; i++ ) {
|
|
|
|
|
lines[i] = null;
|
|
|
|
|
lineLens[i] = 0;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Need to make a copy because we mutate the characters.
|
|
|
|
|
char[] realText = value;
|
|
|
|
|
MakeWrapCopy();
|
|
|
|
|
|
|
|
|
|
int linesCount = 0;
|
|
|
|
|
for( int index = 0; index < totalChars; index += lineSize ) {
|
|
|
|
|
if( value[index] == '\0' )
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
int lineEnd = index + (lineSize - 1), nextLine = lineEnd + 1;
|
|
|
|
|
linesCount++;
|
|
|
|
|
|
|
|
|
|
// Do we need word wrapping?
|
|
|
|
|
bool needWrap = !IsWrapper( value[lineEnd] )
|
|
|
|
|
&& nextLine < totalChars && !IsWrapper( value[nextLine] );
|
|
|
|
|
int wrappedLen = needWrap ? WrapLine( index, lineSize ) : lineSize;
|
|
|
|
|
|
|
|
|
|
// Calculate the maximum size of this line
|
|
|
|
|
int lineLen = lineSize;
|
|
|
|
|
for( int i = lineEnd; i >= index; i-- ) {
|
|
|
|
|
if( value[i] != '\0' ) break;
|
|
|
|
|
lineLen--;
|
|
|
|
|
}
|
|
|
|
|
lineLens[index / lineSize] = Math.Min( lineLen, wrappedLen );
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Output the used lines
|
|
|
|
|
OutputLines( drawer, ref lines, lineLens,
|
|
|
|
|
linesCount, lineSize, totalChars );
|
|
|
|
|
value = realText;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void MakeWrapCopy() {
|
|
|
|
|
int len = Length;
|
|
|
|
|
for( int i = 0; i < len; i++ )
|
|
|
|
|
wrap[i] = value[i];
|
|
|
|
|
|
|
|
|
|
for( int i = len; i < capacity; i++ )
|
|
|
|
|
wrap[i] = '\0';
|
|
|
|
|
value = wrap;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void OutputLines( IDrawer2D drawer, ref string[] lines, int[] lineLens,
|
|
|
|
|
int linesCount, int lineSize, int totalChars ) {
|
|
|
|
|
for( int i = 0; i < totalChars; i++ ) {
|
|
|
|
|
if( value[i] == '\0' ) value[i] = ' ';
|
|
|
|
|
}
|
|
|
|
|
// convert %0-f to &0-f for colour preview.
|
|
|
|
|
for( int i = 0; i < totalChars - 1; i++ ) {
|
|
|
|
|
if( value[i] == '%' && drawer.ValidColour( value[i + 1] ) )
|
|
|
|
|
value[i] = '&';
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
for( int i = 0; i < Math.Max( 1, linesCount ); i++ )
|
|
|
|
|
lines[i] = new String( value, i * lineSize, lineLens[i] );
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
int WrapLine( int index, int lineSize ) {
|
|
|
|
|
int lineEnd = index + (lineSize - 1);
|
|
|
|
|
// wrap - but we don't want to wrap if the entire line is filled.
|
|
|
|
|
for( int i = lineEnd; i >= index + 1; i-- ) {
|
|
|
|
|
if( IsWrapper( value[i] ) ) {
|
|
|
|
|
for( int j = lineEnd; j >= i + 1; j-- ) {
|
|
|
|
|
InsertAt( index + lineSize, value[j] );
|
|
|
|
|
value[j] = ' ';
|
|
|
|
|
}
|
|
|
|
|
return (i + 1) - index;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
return lineSize;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
bool IsWrapper( char c ) {
|
|
|
|
|
return c == '\0' || c == ' ' || c == '-' || c == '>'
|
|
|
|
|
|| c == '<' || c == '/' || c == '\\';
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public void MakeCoords( int index, int[] partLens, out int x, out int y ) {
|
|
|
|
|
if( index == -1 ) index = Int32.MaxValue;
|
|
|
|
|
int total = 0; x = -1; y = 0;
|
|
|
|
|
|
|
|
|
|
for( int yy = 0; yy < partLens.Length; yy++ ) {
|
|
|
|
|
if( partLens[yy] == 0 ) break;
|
|
|
|
|
|
|
|
|
|
y = yy;
|
|
|
|
|
if( index < total + partLens[yy] ) {
|
|
|
|
|
x = index - total; break;
|
|
|
|
|
}
|
|
|
|
|
total += partLens[yy];
|
|
|
|
|
}
|
|
|
|
|
if( x == -1 ) x = partLens[y];
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public int GetBackLength( int index ) {
|
|
|
|
|
if( index <= 0 ) return 0;
|
|
|
|
|
int start = index;
|
|
|
|
|
|
|
|
|
|
bool lookingSpace = !Char.IsLetterOrDigit( value[index] );
|
|
|
|
|
// go back to the end of the previous word
|
|
|
|
|
if( lookingSpace ) {
|
|
|
|
|
while( index > 0 && !Char.IsLetterOrDigit( value[index] ) )
|
|
|
|
|
index--;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// go back to the start of the current word
|
|
|
|
|
while( index > 0 && Char.IsLetterOrDigit( value[index] ) )
|
|
|
|
|
index--;
|
|
|
|
|
return (start - index);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
public int GetForwardLength( int index ) {
|
|
|
|
|
if( index == -1 ) return 0;
|
|
|
|
|
int start = index;
|
|
|
|
|
|
|
|
|
|
bool lookingLetter = Char.IsLetterOrDigit( value[index] );
|
|
|
|
|
// go forward to the end of the current word
|
|
|
|
|
if( lookingLetter ) {
|
|
|
|
|
while( index < Length && Char.IsLetterOrDigit( value[index] ) )
|
|
|
|
|
index++;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// go forward to the start of the next word
|
|
|
|
|
while( index < Length && !Char.IsLetterOrDigit( value[index] ) )
|
|
|
|
|
index++;
|
|
|
|
|
return index - start;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|