pl.kotcrab.vis.ui.widget.VisTextArea Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of vis-ui Show documentation
Show all versions of vis-ui Show documentation
Flat UI skin for LibGDX scene2d.ui
/*******************************************************************************
* Copyright 2014 Pawel Pastuszak
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
******************************************************************************/
package pl.kotcrab.vis.ui.widget;
import com.badlogic.gdx.Gdx;
import com.badlogic.gdx.Input;
import com.badlogic.gdx.graphics.g2d.Batch;
import com.badlogic.gdx.graphics.g2d.BitmapFont;
import com.badlogic.gdx.scenes.scene2d.InputEvent;
import com.badlogic.gdx.scenes.scene2d.InputListener;
import com.badlogic.gdx.scenes.scene2d.Stage;
import com.badlogic.gdx.scenes.scene2d.utils.Drawable;
import com.badlogic.gdx.utils.IntArray;
public class VisTextArea extends VisTextField {
// This class was copied from LibGDX, few lines were changed.
/** Array storing lines breaks positions **/
IntArray linesBreak;
/** Last text processed. This attribute is used to avoid unnecessary computations while calculating offsets **/
private String lastText;
/** Current line for the cursor **/
int cursorLine;
/** Index of the first line showed by the text area **/
int firstLineShowing;
/** Number of lines showed by the text area **/
private int linesShowing;
/** Variable to maintain the x offset of the cursor when moving up and down. If it's set to -1, the offset is reset **/
float moveOffset;
private float prefRows;
public VisTextArea () {
super();
}
public VisTextArea (String text, String styleName) {
super(text, styleName);
}
public VisTextArea (String text, VisTextFieldStyle style) {
super(text, style);
}
public VisTextArea (String text) {
super(text);
}
@Override
protected void initialize () {
super.initialize();
writeEnters = true;
linesBreak = new IntArray();
cursorLine = 0;
firstLineShowing = 0;
moveOffset = -1;
linesShowing = 0;
}
@Override
protected int letterUnderCursor (float x) {
if (linesBreak.size > 0) {
if (cursorLine * 2 >= linesBreak.size) {
return text.length();
} else {
int start = linesBreak.items[cursorLine * 2];
int end = linesBreak.items[cursorLine * 2 + 1];
int i = start;
boolean found = false;
while (i <= end && !found) {
if (glyphPositions.items[i] - glyphPositions.items[start] > x) {
found = true;
} else {
i++;
}
}
return Math.max(0, i - 1);
}
} else {
return 0;
}
}
/** Sets the preferred number of rows (lines) for this text area. Used to calculate preferred height */
public void setPrefRows (float prefRows) {
this.prefRows = prefRows;
}
@Override
public float getPrefHeight () {
if (prefRows <= 0) {
return super.getPrefHeight();
} else {
float prefHeight = textHeight * prefRows;
if (style.background != null) {
prefHeight = Math.max(prefHeight + style.background.getBottomHeight() + style.background.getTopHeight(),
style.background.getMinHeight());
}
return prefHeight;
}
}
/** Returns total number of lines that the text occupies **/
public int getLines () {
return linesBreak.size / 2 + (newLineAtEnd() ? 1 : 0);
}
/** Returns if there's a new line at then end of the text **/
public boolean newLineAtEnd () {
return text.length() != 0
&& (text.charAt(text.length() - 1) == ENTER_ANDROID || text.charAt(text.length() - 1) == ENTER_DESKTOP);
}
/** Moves the cursor to the given number line **/
public void moveCursorLine (int line) {
if (line < 0) {
cursorLine = 0;
cursor = 0;
moveOffset = -1;
} else if (line >= getLines()) {
int newLine = getLines() - 1;
cursor = text.length();
if (line > getLines() || newLine == cursorLine) {
moveOffset = -1;
}
cursorLine = newLine;
} else if (line != cursorLine) {
if (moveOffset < 0) {
moveOffset = linesBreak.size <= cursorLine * 2 ? 0 : glyphPositions.get(cursor)
- glyphPositions.get(linesBreak.get(cursorLine * 2));
}
cursorLine = line;
cursor = cursorLine * 2 >= linesBreak.size ? text.length() : linesBreak.get(cursorLine * 2);
while (cursor < text.length() && cursor <= linesBreak.get(cursorLine * 2 + 1) - 1
&& glyphPositions.get(cursor) - glyphPositions.get(linesBreak.get(cursorLine * 2)) < moveOffset) {
cursor++;
}
showCursor();
}
}
/** Updates the current line, checking the cursor position in the text **/
void updateCurrentLine () {
int index = calculateCurrentLineIndex(cursor);
int line = index / 2;
// Special case when cursor moves to the beginning of the line from the end of another and a word
// wider than the box
if (index % 2 == 0 || index + 1 >= linesBreak.size || cursor != linesBreak.items[index]
|| linesBreak.items[index + 1] != linesBreak.items[index]) {
if (line < linesBreak.size / 2 || text.length() == 0 || text.charAt(text.length() - 1) == ENTER_ANDROID
|| text.charAt(text.length() - 1) == ENTER_DESKTOP) {
cursorLine = line;
}
}
}
/** Scroll the text area to show the line of the cursor **/
void showCursor () {
updateCurrentLine();
if (cursorLine != firstLineShowing) {
int step = cursorLine >= firstLineShowing ? 1 : -1;
while (firstLineShowing > cursorLine || firstLineShowing + linesShowing - 1 < cursorLine) {
firstLineShowing += step;
}
}
}
/** Calculates the text area line for the given cursor position **/
private int calculateCurrentLineIndex (int cursor) {
int index = 0;
while (index < linesBreak.size && cursor > linesBreak.items[index]) {
index++;
}
return index;
}
// OVERRIDE from TextField
@Override
protected void sizeChanged () {
// The number of lines showed must be updated whenever the height is updated
BitmapFont font = style.font;
Drawable background = style.background;
float availableHeight = getHeight() - (background == null ? 0 : background.getBottomHeight() + background.getTopHeight());
linesShowing = (int)Math.floor(availableHeight / font.getLineHeight());
}
@Override
protected float getTextY (BitmapFont font, Drawable background) {
float textY = getHeight();
if (background != null) {
textY = (int)(textY - background.getTopHeight());
}
return textY;
}
@Override
protected void drawSelection (Drawable selection, Batch batch, BitmapFont font, float x, float y) {
int i = firstLineShowing * 2;
float offsetY = 0;
int minIndex = Math.min(cursor, selectionStart);
int maxIndex = Math.max(cursor, selectionStart);
while (i + 1 < linesBreak.size && i < (firstLineShowing + linesShowing) * 2) {
int lineStart = linesBreak.get(i);
int lineEnd = linesBreak.get(i + 1);
if (!((minIndex < lineStart && minIndex < lineEnd && maxIndex < lineStart && maxIndex < lineEnd) || (minIndex > lineStart
&& minIndex > lineEnd && maxIndex > lineStart && maxIndex > lineEnd))) {
int start = Math.max(linesBreak.get(i), minIndex);
int end = Math.min(linesBreak.get(i + 1), maxIndex);
float selectionX = glyphPositions.get(start) - glyphPositions.get(linesBreak.get(i));
float selectionWidth = glyphPositions.get(end) - glyphPositions.get(start);
selection.draw(batch, x + selectionX, y - textHeight - font.getDescent() - offsetY, selectionWidth,
font.getLineHeight());
}
offsetY += font.getLineHeight();
i += 2;
}
}
@Override
protected void drawText (Batch batch, BitmapFont font, float x, float y) {
float offsetY = 0;
for (int i = firstLineShowing * 2; i < (firstLineShowing + linesShowing) * 2 && i < linesBreak.size; i += 2) {
font.draw(batch, displayText, x, y + offsetY, linesBreak.items[i], linesBreak.items[i + 1]);
offsetY -= font.getLineHeight();
}
}
@Override
protected void drawCursor (Drawable cursorPatch, Batch batch, BitmapFont font, float x, float y) {
float textOffset = cursor >= glyphPositions.size || cursorLine * 2 >= linesBreak.size ? 0 : glyphPositions.get(cursor)
- glyphPositions.get(linesBreak.items[cursorLine * 2]);
cursorPatch.draw(batch, x + textOffset,
y - font.getDescent() / 2 - (cursorLine - firstLineShowing + 1) * font.getLineHeight(), cursorPatch.getMinWidth(),
font.getLineHeight());
}
@Override
protected void calculateOffsets () {
super.calculateOffsets();
if (!this.text.equals(lastText)) {
this.lastText = text;
BitmapFont.TextBounds bounds = new BitmapFont.TextBounds();
BitmapFont font = style.font;
float maxWidthLine = this.getWidth()
- (style.background != null ? style.background.getLeftWidth() + style.background.getRightWidth() : 0);
linesBreak.clear();
int lineStart = 0;
int lastSpace = 0;
char lastCharacter;
for (int i = 0; i < text.length(); i++) {
lastCharacter = text.charAt(i);
if (lastCharacter == ENTER_DESKTOP || lastCharacter == ENTER_ANDROID) {
linesBreak.add(lineStart);
linesBreak.add(i);
lineStart = i + 1;
} else {
lastSpace = (continueCursor(i, 0) ? lastSpace : i);
font.getBounds(text, lineStart, i + 1, bounds);
if (bounds.width > maxWidthLine) {
if (lineStart >= lastSpace) {
lastSpace = i - 1;
}
linesBreak.add(lineStart);
linesBreak.add(lastSpace + 1);
lineStart = lastSpace + 1;
lastSpace = lineStart;
}
}
}
// Add last line
if (lineStart < text.length()) {
linesBreak.add(lineStart);
linesBreak.add(text.length());
}
showCursor();
}
}
@Override
protected InputListener createInputListener () {
return new TextAreaListener();
}
@Override
public void setSelection (int selectionStart, int selectionEnd) {
super.setSelection(selectionStart, selectionEnd);
updateCurrentLine();
}
@Override
protected void moveCursor (boolean forward, boolean jump) {
int count = forward ? 1 : -1;
int index = (cursorLine * 2) + count;
if (index >= 0 && index + 1 < linesBreak.size && linesBreak.items[index] == cursor && linesBreak.items[index + 1] == cursor) {
cursorLine += count;
if (jump) {
super.moveCursor(forward, jump);
}
showCursor();
} else {
super.moveCursor(forward, jump);
}
updateCurrentLine();
}
@Override
protected boolean continueCursor (int index, int offset) {
int pos = calculateCurrentLineIndex(index + offset);
return super.continueCursor(index, offset)
&& (pos < 0 || pos >= linesBreak.size || (linesBreak.items[pos + 1] != index) || (linesBreak.items[pos + 1] == linesBreak.items[pos + 2]));
}
public int getCursorLine () {
return cursorLine;
}
public int getFirstLineShowing () {
return firstLineShowing;
}
public int getLinesShowing () {
return linesShowing;
}
/** Input listener for the text area **/
public class TextAreaListener extends TextFieldClickListener {
@Override
protected void setCursorPosition (float x, float y) {
moveOffset = -1;
Drawable background = style.background;
BitmapFont font = style.font;
float height = getHeight();
if (background != null) {
height -= background.getTopHeight();
x -= background.getLeftWidth();
}
x = Math.max(0, x);
if (background != null) {
y -= background.getTopHeight();
}
cursorLine = (int)Math.floor((height - y) / font.getLineHeight()) + firstLineShowing;
cursorLine = Math.max(0, Math.min(cursorLine, getLines() - 1));
super.setCursorPosition(x, y);
updateCurrentLine();
}
@Override
public boolean keyDown (InputEvent event, int keycode) {
super.keyDown(event, keycode);
Stage stage = getStage();
if (stage != null && stage.getKeyboardFocus() == VisTextArea.this) {
boolean repeat = false;
boolean shift = Gdx.input.isKeyPressed(Input.Keys.SHIFT_LEFT) || Gdx.input.isKeyPressed(Input.Keys.SHIFT_RIGHT);
if (keycode == Input.Keys.DOWN) {
if (shift) {
if (!hasSelection) {
selectionStart = cursor;
hasSelection = true;
}
} else {
clearSelection();
}
moveCursorLine(cursorLine + 1);
repeat = true;
} else if (keycode == Input.Keys.UP) {
if (shift) {
if (!hasSelection) {
selectionStart = cursor;
hasSelection = true;
}
} else {
clearSelection();
}
moveCursorLine(cursorLine - 1);
repeat = true;
} else {
moveOffset = -1;
}
if (repeat) {
scheduleKeyRepeatTask(keycode);
}
showCursor();
return true;
}
return false;
}
@Override
public boolean keyTyped (InputEvent event, char character) {
boolean result = super.keyTyped(event, character);
showCursor();
return result;
}
@Override
protected void goHome (boolean jump) {
if (jump) {
cursor = 0;
} else if (cursorLine * 2 < linesBreak.size) {
cursor = linesBreak.get(cursorLine * 2);
}
}
@Override
protected void goEnd (boolean jump) {
if (jump || cursorLine >= getLines()) {
cursor = text.length();
} else if (cursorLine * 2 + 1 < linesBreak.size) {
cursor = linesBreak.get(cursorLine * 2 + 1);
}
}
}
}