mirror of
https://github.com/SquidDev-CC/CC-Tweaked
synced 2025-10-24 10:27:38 +00:00
Backport 1.15's terminal rendering code (#412)
This is a backport of 1.15's terminal rendering code with some further improvements. This duplicates a fair bit of code, and is much more efficient. I expect the work done in #409 will supersede this, but that's unlikely to make its way into the next release so it's worth getting this in for now. - Refactor a lot of common terminal code into `FixedWithFontRenderer`. This shouldn't change any behaviour, but makes a lot of our terminal renderers (printed pages, terminals, monitors) a lot cleaner. - Terminal rendering is done using a single mode/vertex format. Rather than drawing an untextured quad for the background colours, we use an entirely white piece of the terminal font. This allows us to batch draws together more elegantly. - Some minor optimisations: - Skip rendering `"\0"` and `" "` characters. These characters occur pretty often, especially on blank monitors and, as the font is empty here, it is safe to skip them. - Batch together adjacent background cells of the same colour. Again, most terminals will have large runs of the same colour, so this is a worthwhile optimisation. These optimisations do mean that terminal performance is no longer consistent as "noisy" terminals will have worse performance. This is annoying, but still worthwhile. - Switch monitor rendering over to use VBOs. We also add a config option to switch between rendering backends. By default we'll choose the best one compatible with your GPU, but there is a config option to switch between VBOS (reasonable performance) and display lists (bad). When benchmarking 30 full-sized monitors rendering a static image, this improves my FPS[^1] from 7 to 95. This is obviously an extreme case - monitor updates are still slow, and so more frequently updating screens will still be less than stellar. [^1]: My graphics card is an Intel HD Graphics 520. Obviously numbers will vary.
This commit is contained in:
@@ -46,6 +46,7 @@ import dan200.computercraft.shared.peripheral.modem.wired.ItemCable;
|
|||||||
import dan200.computercraft.shared.peripheral.modem.wireless.BlockAdvancedModem;
|
import dan200.computercraft.shared.peripheral.modem.wireless.BlockAdvancedModem;
|
||||||
import dan200.computercraft.shared.peripheral.modem.wireless.ItemAdvancedModem;
|
import dan200.computercraft.shared.peripheral.modem.wireless.ItemAdvancedModem;
|
||||||
import dan200.computercraft.shared.peripheral.modem.wireless.WirelessNetwork;
|
import dan200.computercraft.shared.peripheral.modem.wireless.WirelessNetwork;
|
||||||
|
import dan200.computercraft.shared.peripheral.monitor.MonitorRenderer;
|
||||||
import dan200.computercraft.shared.pocket.items.ItemPocketComputer;
|
import dan200.computercraft.shared.pocket.items.ItemPocketComputer;
|
||||||
import dan200.computercraft.shared.pocket.peripherals.PocketModem;
|
import dan200.computercraft.shared.pocket.peripherals.PocketModem;
|
||||||
import dan200.computercraft.shared.pocket.peripherals.PocketSpeaker;
|
import dan200.computercraft.shared.pocket.peripherals.PocketSpeaker;
|
||||||
@@ -135,6 +136,7 @@ public class ComputerCraft
|
|||||||
public static int modem_rangeDuringStorm = 64;
|
public static int modem_rangeDuringStorm = 64;
|
||||||
public static int modem_highAltitudeRangeDuringStorm = 384;
|
public static int modem_highAltitudeRangeDuringStorm = 384;
|
||||||
public static int maxNotesPerTick = 8;
|
public static int maxNotesPerTick = 8;
|
||||||
|
public static MonitorRenderer monitorRenderer = MonitorRenderer.BEST;
|
||||||
|
|
||||||
public static boolean turtlesNeedFuel = true;
|
public static boolean turtlesNeedFuel = true;
|
||||||
public static int turtleFuelLimit = 20000;
|
public static int turtleFuelLimit = 20000;
|
||||||
|
@@ -5,195 +5,324 @@
|
|||||||
*/
|
*/
|
||||||
package dan200.computercraft.client.gui;
|
package dan200.computercraft.client.gui;
|
||||||
|
|
||||||
|
import dan200.computercraft.client.FrameInfo;
|
||||||
|
import dan200.computercraft.core.terminal.Terminal;
|
||||||
import dan200.computercraft.core.terminal.TextBuffer;
|
import dan200.computercraft.core.terminal.TextBuffer;
|
||||||
|
import dan200.computercraft.shared.util.Colour;
|
||||||
import dan200.computercraft.shared.util.Palette;
|
import dan200.computercraft.shared.util.Palette;
|
||||||
import net.minecraft.client.Minecraft;
|
import net.minecraft.client.Minecraft;
|
||||||
import net.minecraft.client.renderer.BufferBuilder;
|
import net.minecraft.client.renderer.BufferBuilder;
|
||||||
import net.minecraft.client.renderer.GlStateManager;
|
import net.minecraft.client.renderer.GlStateManager;
|
||||||
import net.minecraft.client.renderer.Tessellator;
|
import net.minecraft.client.renderer.Tessellator;
|
||||||
import net.minecraft.client.renderer.texture.TextureManager;
|
|
||||||
import net.minecraft.client.renderer.vertex.DefaultVertexFormats;
|
import net.minecraft.client.renderer.vertex.DefaultVertexFormats;
|
||||||
|
import net.minecraft.client.renderer.vertex.VertexFormat;
|
||||||
import net.minecraft.util.ResourceLocation;
|
import net.minecraft.util.ResourceLocation;
|
||||||
import org.lwjgl.opengl.GL11;
|
import org.lwjgl.opengl.GL11;
|
||||||
|
|
||||||
import java.util.Arrays;
|
import javax.annotation.Nonnull;
|
||||||
|
import javax.annotation.Nullable;
|
||||||
|
|
||||||
public final class FixedWidthFontRenderer
|
public final class FixedWidthFontRenderer
|
||||||
{
|
{
|
||||||
private static final ResourceLocation FONT = new ResourceLocation( "computercraft", "textures/gui/term_font.png" );
|
private static final ResourceLocation FONT = new ResourceLocation( "computercraft", "textures/gui/term_font.png" );
|
||||||
public static final ResourceLocation BACKGROUND = new ResourceLocation( "computercraft", "textures/gui/term_background.png" );
|
|
||||||
|
/**
|
||||||
|
* Like {@link DefaultVertexFormats#POSITION_TEX_COLOR}, but flipped. This is backported from 1.15, hence the
|
||||||
|
* custom format.
|
||||||
|
*/
|
||||||
|
public static final VertexFormat POSITION_COLOR_TEX = new VertexFormat();
|
||||||
|
|
||||||
|
static
|
||||||
|
{
|
||||||
|
POSITION_COLOR_TEX.addElement( DefaultVertexFormats.POSITION_3F );
|
||||||
|
POSITION_COLOR_TEX.addElement( DefaultVertexFormats.COLOR_4UB );
|
||||||
|
POSITION_COLOR_TEX.addElement( DefaultVertexFormats.TEX_2F );
|
||||||
|
}
|
||||||
|
|
||||||
public static final int FONT_HEIGHT = 9;
|
public static final int FONT_HEIGHT = 9;
|
||||||
public static final int FONT_WIDTH = 6;
|
public static final int FONT_WIDTH = 6;
|
||||||
|
public static final float WIDTH = 256.0f;
|
||||||
|
|
||||||
private static FixedWidthFontRenderer instance;
|
public static final float BACKGROUND_START = (WIDTH - 6.0f) / WIDTH;
|
||||||
|
public static final float BACKGROUND_END = (WIDTH - 4.0f) / WIDTH;
|
||||||
public static FixedWidthFontRenderer instance()
|
|
||||||
{
|
|
||||||
if( instance != null ) return instance;
|
|
||||||
return instance = new FixedWidthFontRenderer();
|
|
||||||
}
|
|
||||||
|
|
||||||
private final TextureManager m_textureManager;
|
|
||||||
|
|
||||||
private FixedWidthFontRenderer()
|
private FixedWidthFontRenderer()
|
||||||
{
|
{
|
||||||
m_textureManager = Minecraft.getMinecraft().getTextureManager();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
private static void greyscaleify( double[] rgb )
|
private static float toGreyscale( double[] rgb )
|
||||||
{
|
{
|
||||||
Arrays.fill( rgb, (rgb[0] + rgb[1] + rgb[2]) / 3.0f );
|
return (float) ((rgb[0] + rgb[1] + rgb[2]) / 3);
|
||||||
}
|
}
|
||||||
|
|
||||||
private void drawChar( BufferBuilder renderer, double x, double y, int index, int color, Palette p, boolean greyscale )
|
private static int getColour( char c )
|
||||||
{
|
{
|
||||||
|
int i = "0123456789abcdef".indexOf( c );
|
||||||
|
return i < 0 ? 0 : 15 - i;
|
||||||
|
}
|
||||||
|
|
||||||
|
private static void drawChar( BufferBuilder buffer, float x, float y, int index, float r, float g, float b )
|
||||||
|
{
|
||||||
|
// Short circuit to avoid the common case - the texture should be blank here after all.
|
||||||
|
if( index == '\0' || index == ' ' ) return;
|
||||||
|
|
||||||
int column = index % 16;
|
int column = index % 16;
|
||||||
int row = index / 16;
|
int row = index / 16;
|
||||||
|
|
||||||
double[] colour = p.getColour( 15 - color );
|
|
||||||
if( greyscale )
|
|
||||||
{
|
|
||||||
greyscaleify( colour );
|
|
||||||
}
|
|
||||||
float r = (float) colour[0];
|
|
||||||
float g = (float) colour[1];
|
|
||||||
float b = (float) colour[2];
|
|
||||||
|
|
||||||
int xStart = 1 + column * (FONT_WIDTH + 2);
|
int xStart = 1 + column * (FONT_WIDTH + 2);
|
||||||
int yStart = 1 + row * (FONT_HEIGHT + 2);
|
int yStart = 1 + row * (FONT_HEIGHT + 2);
|
||||||
|
|
||||||
renderer.pos( x, y, 0.0 ).tex( xStart / 256.0, yStart / 256.0 ).color( r, g, b, 1.0f ).endVertex();
|
buffer.pos( x, y, 0f ).color( r, g, b, 1.0f ).tex( xStart / WIDTH, yStart / WIDTH ).endVertex();
|
||||||
renderer.pos( x, y + FONT_HEIGHT, 0.0 ).tex( xStart / 256.0, (yStart + FONT_HEIGHT) / 256.0 ).color( r, g, b, 1.0f ).endVertex();
|
buffer.pos( x, y + FONT_HEIGHT, 0f ).color( r, g, b, 1.0f ).tex( xStart / WIDTH, (yStart + FONT_HEIGHT) / WIDTH ).endVertex();
|
||||||
renderer.pos( x + FONT_WIDTH, y, 0.0 ).tex( (xStart + FONT_WIDTH) / 256.0, yStart / 256.0 ).color( r, g, b, 1.0f ).endVertex();
|
buffer.pos( x + FONT_WIDTH, y, 0f ).color( r, g, b, 1.0f ).tex( (xStart + FONT_WIDTH) / WIDTH, yStart / WIDTH ).endVertex();
|
||||||
renderer.pos( x + FONT_WIDTH, y, 0.0 ).tex( (xStart + FONT_WIDTH) / 256.0, yStart / 256.0 ).color( r, g, b, 1.0f ).endVertex();
|
buffer.pos( x + FONT_WIDTH, y, 0f ).color( r, g, b, 1.0f ).tex( (xStart + FONT_WIDTH) / WIDTH, yStart / WIDTH ).endVertex();
|
||||||
renderer.pos( x, y + FONT_HEIGHT, 0.0 ).tex( xStart / 256.0, (yStart + FONT_HEIGHT) / 256.0 ).color( r, g, b, 1.0f ).endVertex();
|
buffer.pos( x, y + FONT_HEIGHT, 0f ).color( r, g, b, 1.0f ).tex( xStart / WIDTH, (yStart + FONT_HEIGHT) / WIDTH ).endVertex();
|
||||||
renderer.pos( x + FONT_WIDTH, y + FONT_HEIGHT, 0.0 ).tex( (xStart + FONT_WIDTH) / 256.0, (yStart + FONT_HEIGHT) / 256.0 ).color( r, g, b, 1.0f ).endVertex();
|
buffer.pos( x + FONT_WIDTH, y + FONT_HEIGHT, 0f ).color( r, g, b, 1.0f ).tex( (xStart + FONT_WIDTH) / WIDTH, (yStart + FONT_HEIGHT) / WIDTH ).endVertex();
|
||||||
}
|
}
|
||||||
|
|
||||||
private void drawQuad( BufferBuilder renderer, double x, double y, int color, double width, Palette p, boolean greyscale )
|
private static void drawQuad( BufferBuilder buffer, float x, float y, float width, float height, float r, float g, float b )
|
||||||
{
|
{
|
||||||
double[] colour = p.getColour( 15 - color );
|
buffer.pos( x, y, 0 ).color( r, g, b, 1.0f ).tex( BACKGROUND_START, BACKGROUND_START ).endVertex();
|
||||||
|
buffer.pos( x, y + height, 0 ).color( r, g, b, 1.0f ).tex( BACKGROUND_START, BACKGROUND_END ).endVertex();
|
||||||
|
buffer.pos( x + width, y, 0 ).color( r, g, b, 1.0f ).tex( BACKGROUND_END, BACKGROUND_START ).endVertex();
|
||||||
|
buffer.pos( x + width, y, 0 ).color( r, g, b, 1.0f ).tex( BACKGROUND_END, BACKGROUND_START ).endVertex();
|
||||||
|
buffer.pos( x, y + height, 0 ).color( r, g, b, 1.0f ).tex( BACKGROUND_START, BACKGROUND_END ).endVertex();
|
||||||
|
buffer.pos( x + width, y + height, 0 ).color( r, g, b, 1.0f ).tex( BACKGROUND_END, BACKGROUND_END ).endVertex();
|
||||||
|
}
|
||||||
|
|
||||||
|
private static void drawQuad( BufferBuilder buffer, float x, float y, float width, float height, Palette palette, boolean greyscale, char colourIndex )
|
||||||
|
{
|
||||||
|
double[] colour = palette.getColour( getColour( colourIndex ) );
|
||||||
|
float r, g, b;
|
||||||
if( greyscale )
|
if( greyscale )
|
||||||
{
|
{
|
||||||
greyscaleify( colour );
|
r = g = b = toGreyscale( colour );
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
r = (float) colour[0];
|
||||||
|
g = (float) colour[1];
|
||||||
|
b = (float) colour[2];
|
||||||
}
|
}
|
||||||
float r = (float) colour[0];
|
|
||||||
float g = (float) colour[1];
|
|
||||||
float b = (float) colour[2];
|
|
||||||
|
|
||||||
renderer.pos( x, y, 0.0 ).color( r, g, b, 1.0f ).endVertex();
|
drawQuad( buffer, x, y, width, height, r, g, b );
|
||||||
renderer.pos( x, y + FONT_HEIGHT, 0.0 ).color( r, g, b, 1.0f ).endVertex();
|
|
||||||
renderer.pos( x + width, y, 0.0 ).color( r, g, b, 1.0f ).endVertex();
|
|
||||||
renderer.pos( x + width, y, 0.0 ).color( r, g, b, 1.0f ).endVertex();
|
|
||||||
renderer.pos( x, y + FONT_HEIGHT, 0.0 ).color( r, g, b, 1.0f ).endVertex();
|
|
||||||
renderer.pos( x + width, y + FONT_HEIGHT, 0.0 ).color( r, g, b, 1.0f ).endVertex();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
private boolean isGreyScale( int colour )
|
private static void drawBackground(
|
||||||
|
@Nonnull BufferBuilder renderer, float x, float y,
|
||||||
|
@Nonnull TextBuffer backgroundColour, @Nonnull Palette palette, boolean greyscale,
|
||||||
|
float leftMarginSize, float rightMarginSize, float height
|
||||||
|
)
|
||||||
{
|
{
|
||||||
return colour == 0 || colour == 15 || colour == 7 || colour == 8;
|
if( leftMarginSize > 0 )
|
||||||
}
|
{
|
||||||
|
drawQuad( renderer, x - leftMarginSize, y, leftMarginSize, height, palette, greyscale, backgroundColour.charAt( 0 ) );
|
||||||
|
}
|
||||||
|
|
||||||
public void drawStringBackgroundPart( int x, int y, TextBuffer backgroundColour, double leftMarginSize, double rightMarginSize, boolean greyScale, Palette p )
|
if( rightMarginSize > 0 )
|
||||||
{
|
|
||||||
// Draw the quads
|
|
||||||
Tessellator tessellator = Tessellator.getInstance();
|
|
||||||
BufferBuilder renderer = tessellator.getBuffer();
|
|
||||||
renderer.begin( GL11.GL_TRIANGLES, DefaultVertexFormats.POSITION_COLOR );
|
|
||||||
if( leftMarginSize > 0.0 )
|
|
||||||
{
|
{
|
||||||
int colour1 = "0123456789abcdef".indexOf( backgroundColour.charAt( 0 ) );
|
drawQuad( renderer, x + backgroundColour.length() * FONT_WIDTH, y, rightMarginSize, height, palette, greyscale, backgroundColour.charAt( backgroundColour.length() - 1 ) );
|
||||||
if( colour1 < 0 || (greyScale && !isGreyScale( colour1 )) )
|
|
||||||
{
|
|
||||||
colour1 = 15;
|
|
||||||
}
|
|
||||||
drawQuad( renderer, x - leftMarginSize, y, colour1, leftMarginSize, p, greyScale );
|
|
||||||
}
|
|
||||||
if( rightMarginSize > 0.0 )
|
|
||||||
{
|
|
||||||
int colour2 = "0123456789abcdef".indexOf( backgroundColour.charAt( backgroundColour.length() - 1 ) );
|
|
||||||
if( colour2 < 0 || (greyScale && !isGreyScale( colour2 )) )
|
|
||||||
{
|
|
||||||
colour2 = 15;
|
|
||||||
}
|
|
||||||
drawQuad( renderer, x + backgroundColour.length() * FONT_WIDTH, y, colour2, rightMarginSize, p, greyScale );
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Batch together runs of identical background cells.
|
||||||
|
int blockStart = 0;
|
||||||
|
char blockColour = '\0';
|
||||||
for( int i = 0; i < backgroundColour.length(); i++ )
|
for( int i = 0; i < backgroundColour.length(); i++ )
|
||||||
{
|
{
|
||||||
int colour = "0123456789abcdef".indexOf( backgroundColour.charAt( i ) );
|
char colourIndex = backgroundColour.charAt( i );
|
||||||
if( colour < 0 || (greyScale && !isGreyScale( colour )) )
|
if( colourIndex == blockColour ) continue;
|
||||||
|
|
||||||
|
if( blockColour != '\0' )
|
||||||
{
|
{
|
||||||
colour = 15;
|
drawQuad( renderer, x + blockStart * FONT_WIDTH, y, FONT_WIDTH * (i - blockStart), height, palette, greyscale, blockColour );
|
||||||
}
|
}
|
||||||
drawQuad( renderer, x + i * FONT_WIDTH, y, colour, FONT_WIDTH, p, greyScale );
|
|
||||||
|
blockColour = colourIndex;
|
||||||
|
blockStart = i;
|
||||||
|
}
|
||||||
|
|
||||||
|
if( blockColour != '\0' )
|
||||||
|
{
|
||||||
|
drawQuad( renderer, x + blockStart * FONT_WIDTH, y, FONT_WIDTH * (backgroundColour.length() - blockStart), height, palette, greyscale, blockColour );
|
||||||
}
|
}
|
||||||
GlStateManager.disableTexture2D();
|
|
||||||
tessellator.draw();
|
|
||||||
GlStateManager.enableTexture2D();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
public void drawStringTextPart( int x, int y, TextBuffer s, TextBuffer textColour, boolean greyScale, Palette p )
|
public static void drawString(
|
||||||
|
@Nonnull BufferBuilder renderer, float x, float y,
|
||||||
|
@Nonnull TextBuffer text, @Nonnull TextBuffer textColour, @Nullable TextBuffer backgroundColour,
|
||||||
|
@Nonnull Palette palette, boolean greyscale, float leftMarginSize, float rightMarginSize
|
||||||
|
)
|
||||||
{
|
{
|
||||||
// Draw the quads
|
if( backgroundColour != null )
|
||||||
Tessellator tessellator = Tessellator.getInstance();
|
|
||||||
BufferBuilder renderer = tessellator.getBuffer();
|
|
||||||
renderer.begin( GL11.GL_TRIANGLES, DefaultVertexFormats.POSITION_TEX_COLOR );
|
|
||||||
for( int i = 0; i < s.length(); i++ )
|
|
||||||
{
|
{
|
||||||
// Switch colour
|
drawBackground( renderer, x, y, backgroundColour, palette, greyscale, leftMarginSize, rightMarginSize, FONT_HEIGHT );
|
||||||
int colour = "0123456789abcdef".indexOf( textColour.charAt( i ) );
|
}
|
||||||
if( colour < 0 || (greyScale && !isGreyScale( colour )) )
|
|
||||||
|
for( int i = 0; i < text.length(); i++ )
|
||||||
|
{
|
||||||
|
double[] colour = palette.getColour( getColour( textColour.charAt( i ) ) );
|
||||||
|
float r, g, b;
|
||||||
|
if( greyscale )
|
||||||
{
|
{
|
||||||
colour = 0;
|
r = g = b = toGreyscale( colour );
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
r = (float) colour[0];
|
||||||
|
g = (float) colour[1];
|
||||||
|
b = (float) colour[2];
|
||||||
}
|
}
|
||||||
|
|
||||||
// Draw char
|
// Draw char
|
||||||
int index = s.charAt( i );
|
int index = text.charAt( i );
|
||||||
if( index < 0 || index > 255 )
|
if( index > 255 ) index = '?';
|
||||||
{
|
drawChar( renderer, x + i * FONT_WIDTH, y, index, r, g, b );
|
||||||
index = '?';
|
|
||||||
}
|
|
||||||
drawChar( renderer, x + i * FONT_WIDTH, y, index, colour, p, greyScale );
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
public static void drawString(
|
||||||
|
float x, float y, @Nonnull TextBuffer text, @Nonnull TextBuffer textColour, @Nullable TextBuffer backgroundColour,
|
||||||
|
@Nonnull Palette palette, boolean greyscale, float leftMarginSize, float rightMarginSize
|
||||||
|
)
|
||||||
|
{
|
||||||
|
bindFont();
|
||||||
|
|
||||||
|
Tessellator tessellator = Tessellator.getInstance();
|
||||||
|
BufferBuilder buffer = tessellator.getBuffer();
|
||||||
|
begin( buffer );
|
||||||
|
drawString( buffer, x, y, text, textColour, backgroundColour, palette, greyscale, leftMarginSize, rightMarginSize );
|
||||||
tessellator.draw();
|
tessellator.draw();
|
||||||
}
|
}
|
||||||
|
|
||||||
public void drawString( TextBuffer s, int x, int y, TextBuffer textColour, TextBuffer backgroundColour, double leftMarginSize, double rightMarginSize, boolean greyScale, Palette p )
|
public static void drawTerminalWithoutCursor(
|
||||||
|
@Nonnull BufferBuilder buffer, float x, float y,
|
||||||
|
@Nonnull Terminal terminal, boolean greyscale,
|
||||||
|
float topMarginSize, float bottomMarginSize, float leftMarginSize, float rightMarginSize
|
||||||
|
)
|
||||||
{
|
{
|
||||||
// Draw background
|
Palette palette = terminal.getPalette();
|
||||||
if( backgroundColour != null )
|
int height = terminal.getHeight();
|
||||||
|
|
||||||
|
// Top and bottom margins
|
||||||
|
drawBackground(
|
||||||
|
buffer, x, y - topMarginSize,
|
||||||
|
terminal.getBackgroundColourLine( 0 ), palette, greyscale,
|
||||||
|
leftMarginSize, rightMarginSize, topMarginSize
|
||||||
|
);
|
||||||
|
|
||||||
|
drawBackground(
|
||||||
|
buffer, x, y + height * FONT_HEIGHT,
|
||||||
|
terminal.getBackgroundColourLine( height - 1 ), palette, greyscale,
|
||||||
|
leftMarginSize, rightMarginSize, bottomMarginSize
|
||||||
|
);
|
||||||
|
|
||||||
|
// The main text
|
||||||
|
for( int i = 0; i < height; i++ )
|
||||||
{
|
{
|
||||||
// Bind the background texture
|
drawString(
|
||||||
m_textureManager.bindTexture( BACKGROUND );
|
buffer, x, y + FixedWidthFontRenderer.FONT_HEIGHT * i,
|
||||||
|
terminal.getLine( i ), terminal.getTextColourLine( i ), terminal.getBackgroundColourLine( i ),
|
||||||
// Draw the quads
|
palette, greyscale, leftMarginSize, rightMarginSize
|
||||||
drawStringBackgroundPart( x, y, backgroundColour, leftMarginSize, rightMarginSize, greyScale, p );
|
);
|
||||||
}
|
|
||||||
|
|
||||||
// Draw text
|
|
||||||
if( s != null && textColour != null )
|
|
||||||
{
|
|
||||||
// Bind the font texture
|
|
||||||
bindFont();
|
|
||||||
|
|
||||||
// Draw the quads
|
|
||||||
drawStringTextPart( x, y, s, textColour, greyScale, p );
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public int getStringWidth( String s )
|
public static void drawCursor(
|
||||||
|
@Nonnull BufferBuilder buffer, float x, float y,
|
||||||
|
@Nonnull Terminal terminal, boolean greyscale
|
||||||
|
)
|
||||||
{
|
{
|
||||||
if( s == null )
|
Palette palette = terminal.getPalette();
|
||||||
|
int width = terminal.getWidth();
|
||||||
|
int height = terminal.getHeight();
|
||||||
|
|
||||||
|
int cursorX = terminal.getCursorX();
|
||||||
|
int cursorY = terminal.getCursorY();
|
||||||
|
if( terminal.getCursorBlink() && cursorX >= 0 && cursorX < width && cursorY >= 0 && cursorY < height && FrameInfo.getGlobalCursorBlink() )
|
||||||
{
|
{
|
||||||
return 0;
|
double[] colour = palette.getColour( 15 - terminal.getTextColour() );
|
||||||
|
float r, g, b;
|
||||||
|
if( greyscale )
|
||||||
|
{
|
||||||
|
r = g = b = toGreyscale( colour );
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
r = (float) colour[0];
|
||||||
|
g = (float) colour[1];
|
||||||
|
b = (float) colour[2];
|
||||||
|
}
|
||||||
|
|
||||||
|
drawChar( buffer, x + cursorX * FONT_WIDTH, y + cursorY * FONT_HEIGHT, '_', r, g, b );
|
||||||
}
|
}
|
||||||
return s.length() * FONT_WIDTH;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
public void bindFont()
|
public static void drawTerminal(
|
||||||
|
@Nonnull BufferBuilder buffer, float x, float y,
|
||||||
|
@Nonnull Terminal terminal, boolean greyscale,
|
||||||
|
float topMarginSize, float bottomMarginSize, float leftMarginSize, float rightMarginSize
|
||||||
|
)
|
||||||
{
|
{
|
||||||
m_textureManager.bindTexture( FONT );
|
drawTerminalWithoutCursor( buffer, x, y, terminal, greyscale, topMarginSize, bottomMarginSize, leftMarginSize, rightMarginSize );
|
||||||
|
drawCursor( buffer, x, y, terminal, greyscale );
|
||||||
|
}
|
||||||
|
|
||||||
|
public static void drawTerminal(
|
||||||
|
float x, float y, @Nonnull Terminal terminal, boolean greyscale,
|
||||||
|
float topMarginSize, float bottomMarginSize, float leftMarginSize, float rightMarginSize
|
||||||
|
)
|
||||||
|
{
|
||||||
|
bindFont();
|
||||||
|
|
||||||
|
Tessellator tessellator = Tessellator.getInstance();
|
||||||
|
BufferBuilder buffer = tessellator.getBuffer();
|
||||||
|
begin( buffer );
|
||||||
|
drawTerminal( buffer, x, y, terminal, greyscale, topMarginSize, bottomMarginSize, leftMarginSize, rightMarginSize );
|
||||||
|
tessellator.draw();
|
||||||
|
}
|
||||||
|
|
||||||
|
public static void drawEmptyTerminal( float x, float y, float width, float height )
|
||||||
|
{
|
||||||
|
bindFont();
|
||||||
|
|
||||||
|
Tessellator tessellator = Tessellator.getInstance();
|
||||||
|
BufferBuilder buffer = tessellator.getBuffer();
|
||||||
|
begin( buffer );
|
||||||
|
|
||||||
|
Colour colour = Colour.Black;
|
||||||
|
drawQuad( buffer, x, y, width, height, colour.getR(), colour.getG(), colour.getB() );
|
||||||
|
|
||||||
|
tessellator.draw();
|
||||||
|
}
|
||||||
|
|
||||||
|
public static void drawBlocker( @Nonnull BufferBuilder buffer, float x, float y, float width, float height )
|
||||||
|
{
|
||||||
|
Colour colour = Colour.Black;
|
||||||
|
drawQuad( buffer, x, y, width, height, colour.getR(), colour.getG(), colour.getB() );
|
||||||
|
}
|
||||||
|
|
||||||
|
public static void drawBlocker( float x, float y, float width, float height )
|
||||||
|
{
|
||||||
|
bindFont();
|
||||||
|
|
||||||
|
Tessellator tessellator = Tessellator.getInstance();
|
||||||
|
BufferBuilder buffer = tessellator.getBuffer();
|
||||||
|
begin( buffer );
|
||||||
|
drawBlocker( buffer, x, y, width, height );
|
||||||
|
tessellator.draw();
|
||||||
|
}
|
||||||
|
|
||||||
|
public static void bindFont()
|
||||||
|
{
|
||||||
|
Minecraft.getMinecraft().getTextureManager().bindTexture( FONT );
|
||||||
GlStateManager.glTexParameteri( GL11.GL_TEXTURE_2D, GL11.GL_TEXTURE_WRAP_S, GL11.GL_CLAMP );
|
GlStateManager.glTexParameteri( GL11.GL_TEXTURE_2D, GL11.GL_TEXTURE_WRAP_S, GL11.GL_CLAMP );
|
||||||
|
|
||||||
|
GlStateManager.enableTexture2D();
|
||||||
|
}
|
||||||
|
|
||||||
|
public static void begin( BufferBuilder buffer )
|
||||||
|
{
|
||||||
|
buffer.begin( GL11.GL_TRIANGLES, POSITION_COLOR_TEX );
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@@ -5,25 +5,18 @@
|
|||||||
*/
|
*/
|
||||||
package dan200.computercraft.client.gui.widgets;
|
package dan200.computercraft.client.gui.widgets;
|
||||||
|
|
||||||
import dan200.computercraft.client.FrameInfo;
|
|
||||||
import dan200.computercraft.client.gui.FixedWidthFontRenderer;
|
import dan200.computercraft.client.gui.FixedWidthFontRenderer;
|
||||||
import dan200.computercraft.core.terminal.Terminal;
|
import dan200.computercraft.core.terminal.Terminal;
|
||||||
import dan200.computercraft.core.terminal.TextBuffer;
|
|
||||||
import dan200.computercraft.shared.computer.core.IComputer;
|
import dan200.computercraft.shared.computer.core.IComputer;
|
||||||
import dan200.computercraft.shared.computer.core.IComputerContainer;
|
import dan200.computercraft.shared.computer.core.IComputerContainer;
|
||||||
import dan200.computercraft.shared.util.Colour;
|
|
||||||
import dan200.computercraft.shared.util.Palette;
|
|
||||||
import net.minecraft.client.Minecraft;
|
import net.minecraft.client.Minecraft;
|
||||||
import net.minecraft.client.gui.GuiScreen;
|
import net.minecraft.client.gui.GuiScreen;
|
||||||
import net.minecraft.client.renderer.GlStateManager;
|
|
||||||
import net.minecraft.util.ChatAllowedCharacters;
|
import net.minecraft.util.ChatAllowedCharacters;
|
||||||
import org.lwjgl.input.Keyboard;
|
import org.lwjgl.input.Keyboard;
|
||||||
import org.lwjgl.input.Mouse;
|
import org.lwjgl.input.Mouse;
|
||||||
|
|
||||||
import java.util.ArrayList;
|
import java.util.ArrayList;
|
||||||
|
|
||||||
import static dan200.computercraft.client.gui.FixedWidthFontRenderer.BACKGROUND;
|
|
||||||
|
|
||||||
public class WidgetTerminal extends Widget
|
public class WidgetTerminal extends Widget
|
||||||
{
|
{
|
||||||
private static final float TERMINATE_TIME = 0.5f;
|
private static final float TERMINATE_TIME = 0.5f;
|
||||||
@@ -41,10 +34,10 @@ public class WidgetTerminal extends Widget
|
|||||||
private boolean m_focus = false;
|
private boolean m_focus = false;
|
||||||
private boolean m_allowFocusLoss = true;
|
private boolean m_allowFocusLoss = true;
|
||||||
|
|
||||||
private int m_leftMargin;
|
private final int leftMargin;
|
||||||
private int m_rightMargin;
|
private final int rightMargin;
|
||||||
private int m_topMargin;
|
private final int topMargin;
|
||||||
private int m_bottomMargin;
|
private final int bottomMargin;
|
||||||
|
|
||||||
private final ArrayList<Integer> m_keysDown = new ArrayList<>();
|
private final ArrayList<Integer> m_keysDown = new ArrayList<>();
|
||||||
|
|
||||||
@@ -58,10 +51,10 @@ public class WidgetTerminal extends Widget
|
|||||||
|
|
||||||
m_computer = computer;
|
m_computer = computer;
|
||||||
|
|
||||||
m_leftMargin = leftMargin;
|
this.leftMargin = leftMargin;
|
||||||
m_rightMargin = rightMargin;
|
this.rightMargin = rightMargin;
|
||||||
m_topMargin = topMargin;
|
this.topMargin = topMargin;
|
||||||
m_bottomMargin = bottomMargin;
|
this.bottomMargin = bottomMargin;
|
||||||
}
|
}
|
||||||
|
|
||||||
public void setAllowFocusLoss( boolean allowFocusLoss )
|
public void setAllowFocusLoss( boolean allowFocusLoss )
|
||||||
@@ -166,8 +159,8 @@ public class WidgetTerminal extends Widget
|
|||||||
Terminal term = computer.getTerminal();
|
Terminal term = computer.getTerminal();
|
||||||
if( term != null )
|
if( term != null )
|
||||||
{
|
{
|
||||||
int charX = (mouseX - (getXPosition() + m_leftMargin)) / FixedWidthFontRenderer.FONT_WIDTH;
|
int charX = (mouseX - (getXPosition() + leftMargin)) / FixedWidthFontRenderer.FONT_WIDTH;
|
||||||
int charY = (mouseY - (getYPosition() + m_topMargin)) / FixedWidthFontRenderer.FONT_HEIGHT;
|
int charY = (mouseY - (getYPosition() + topMargin)) / FixedWidthFontRenderer.FONT_HEIGHT;
|
||||||
charX = Math.min( Math.max( charX, 0 ), term.getWidth() - 1 );
|
charX = Math.min( Math.max( charX, 0 ), term.getWidth() - 1 );
|
||||||
charY = Math.min( Math.max( charY, 0 ), term.getHeight() - 1 );
|
charY = Math.min( Math.max( charY, 0 ), term.getHeight() - 1 );
|
||||||
|
|
||||||
@@ -223,8 +216,8 @@ public class WidgetTerminal extends Widget
|
|||||||
Terminal term = computer.getTerminal();
|
Terminal term = computer.getTerminal();
|
||||||
if( term != null )
|
if( term != null )
|
||||||
{
|
{
|
||||||
int charX = (mouseX - (getXPosition() + m_leftMargin)) / FixedWidthFontRenderer.FONT_WIDTH;
|
int charX = (mouseX - (getXPosition() + leftMargin)) / FixedWidthFontRenderer.FONT_WIDTH;
|
||||||
int charY = (mouseY - (getYPosition() + m_topMargin)) / FixedWidthFontRenderer.FONT_HEIGHT;
|
int charY = (mouseY - (getYPosition() + topMargin)) / FixedWidthFontRenderer.FONT_HEIGHT;
|
||||||
charX = Math.min( Math.max( charX, 0 ), term.getWidth() - 1 );
|
charX = Math.min( Math.max( charX, 0 ), term.getWidth() - 1 );
|
||||||
charY = Math.min( Math.max( charY, 0 ), term.getHeight() - 1 );
|
charY = Math.min( Math.max( charY, 0 ), term.getHeight() - 1 );
|
||||||
|
|
||||||
@@ -339,74 +332,14 @@ public class WidgetTerminal extends Widget
|
|||||||
Terminal terminal = computer != null ? computer.getTerminal() : null;
|
Terminal terminal = computer != null ? computer.getTerminal() : null;
|
||||||
if( terminal != null )
|
if( terminal != null )
|
||||||
{
|
{
|
||||||
// Draw the terminal
|
FixedWidthFontRenderer.drawTerminal(
|
||||||
boolean greyscale = !computer.isColour();
|
startX + topMargin, startY + bottomMargin,
|
||||||
|
terminal, !computer.isColour(), topMargin, bottomMargin, leftMargin, rightMargin
|
||||||
Palette palette = terminal.getPalette();
|
);
|
||||||
|
|
||||||
// Get the data from the terminal first
|
|
||||||
// Unfortunately we have to keep the lock for the whole of drawing, so the text doesn't change under us.
|
|
||||||
FixedWidthFontRenderer fontRenderer = FixedWidthFontRenderer.instance();
|
|
||||||
boolean tblink = m_focus && terminal.getCursorBlink() && FrameInfo.getGlobalCursorBlink();
|
|
||||||
int tw = terminal.getWidth();
|
|
||||||
int th = terminal.getHeight();
|
|
||||||
int tx = terminal.getCursorX();
|
|
||||||
int ty = terminal.getCursorY();
|
|
||||||
|
|
||||||
int x = startX + m_leftMargin;
|
|
||||||
int y = startY + m_topMargin;
|
|
||||||
|
|
||||||
// Draw margins
|
|
||||||
TextBuffer emptyLine = new TextBuffer( ' ', tw );
|
|
||||||
if( m_topMargin > 0 )
|
|
||||||
{
|
|
||||||
fontRenderer.drawString( emptyLine, x, startY, terminal.getTextColourLine( 0 ), terminal.getBackgroundColourLine( 0 ), m_leftMargin, m_rightMargin, greyscale, palette );
|
|
||||||
}
|
|
||||||
if( m_bottomMargin > 0 )
|
|
||||||
{
|
|
||||||
fontRenderer.drawString( emptyLine, x, startY + 2 * m_bottomMargin + (th - 1) * FixedWidthFontRenderer.FONT_HEIGHT, terminal.getTextColourLine( th - 1 ), terminal.getBackgroundColourLine( th - 1 ), m_leftMargin, m_rightMargin, greyscale, palette );
|
|
||||||
}
|
|
||||||
|
|
||||||
// Draw lines
|
|
||||||
for( int line = 0; line < th; line++ )
|
|
||||||
{
|
|
||||||
TextBuffer text = terminal.getLine( line );
|
|
||||||
TextBuffer colour = terminal.getTextColourLine( line );
|
|
||||||
TextBuffer backgroundColour = terminal.getBackgroundColourLine( line );
|
|
||||||
fontRenderer.drawString( text, x, y, colour, backgroundColour, m_leftMargin, m_rightMargin, greyscale, palette );
|
|
||||||
y += FixedWidthFontRenderer.FONT_HEIGHT;
|
|
||||||
}
|
|
||||||
|
|
||||||
if( tblink && tx >= 0 && ty >= 0 && tx < tw && ty < th )
|
|
||||||
{
|
|
||||||
TextBuffer cursor = new TextBuffer( '_', 1 );
|
|
||||||
TextBuffer cursorColour = new TextBuffer( "0123456789abcdef".charAt( terminal.getTextColour() ), 1 );
|
|
||||||
|
|
||||||
fontRenderer.drawString(
|
|
||||||
cursor,
|
|
||||||
x + FixedWidthFontRenderer.FONT_WIDTH * tx,
|
|
||||||
startY + m_topMargin + FixedWidthFontRenderer.FONT_HEIGHT * ty,
|
|
||||||
cursorColour, null,
|
|
||||||
0, 0,
|
|
||||||
greyscale,
|
|
||||||
palette
|
|
||||||
);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
// Draw a black background
|
FixedWidthFontRenderer.drawEmptyTerminal( startX, startY, getWidth(), getHeight() );
|
||||||
mc.getTextureManager().bindTexture( BACKGROUND );
|
|
||||||
Colour black = Colour.Black;
|
|
||||||
GlStateManager.color( black.getR(), black.getG(), black.getB(), 1.0f );
|
|
||||||
try
|
|
||||||
{
|
|
||||||
drawTexturedModalRect( startX, startY, 0, 0, getWidth(), getHeight() );
|
|
||||||
}
|
|
||||||
finally
|
|
||||||
{
|
|
||||||
GlStateManager.color( 1.0f, 1.0f, 1.0f, 1.0f );
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@@ -6,15 +6,12 @@
|
|||||||
package dan200.computercraft.client.render;
|
package dan200.computercraft.client.render;
|
||||||
|
|
||||||
import dan200.computercraft.ComputerCraft;
|
import dan200.computercraft.ComputerCraft;
|
||||||
import dan200.computercraft.client.FrameInfo;
|
|
||||||
import dan200.computercraft.client.gui.FixedWidthFontRenderer;
|
import dan200.computercraft.client.gui.FixedWidthFontRenderer;
|
||||||
import dan200.computercraft.core.terminal.Terminal;
|
import dan200.computercraft.core.terminal.Terminal;
|
||||||
import dan200.computercraft.core.terminal.TextBuffer;
|
|
||||||
import dan200.computercraft.shared.computer.core.ClientComputer;
|
import dan200.computercraft.shared.computer.core.ClientComputer;
|
||||||
import dan200.computercraft.shared.computer.core.ComputerFamily;
|
import dan200.computercraft.shared.computer.core.ComputerFamily;
|
||||||
import dan200.computercraft.shared.pocket.items.ItemPocketComputer;
|
import dan200.computercraft.shared.pocket.items.ItemPocketComputer;
|
||||||
import dan200.computercraft.shared.util.Colour;
|
import dan200.computercraft.shared.util.Colour;
|
||||||
import dan200.computercraft.shared.util.Palette;
|
|
||||||
import net.minecraft.client.Minecraft;
|
import net.minecraft.client.Minecraft;
|
||||||
import net.minecraft.client.renderer.BufferBuilder;
|
import net.minecraft.client.renderer.BufferBuilder;
|
||||||
import net.minecraft.client.renderer.GlStateManager;
|
import net.minecraft.client.renderer.GlStateManager;
|
||||||
@@ -27,7 +24,8 @@ import net.minecraftforge.fml.common.eventhandler.SubscribeEvent;
|
|||||||
import net.minecraftforge.fml.relauncher.Side;
|
import net.minecraftforge.fml.relauncher.Side;
|
||||||
import org.lwjgl.opengl.GL11;
|
import org.lwjgl.opengl.GL11;
|
||||||
|
|
||||||
import static dan200.computercraft.client.gui.FixedWidthFontRenderer.*;
|
import static dan200.computercraft.client.gui.FixedWidthFontRenderer.FONT_HEIGHT;
|
||||||
|
import static dan200.computercraft.client.gui.FixedWidthFontRenderer.FONT_WIDTH;
|
||||||
import static dan200.computercraft.client.gui.GuiComputer.*;
|
import static dan200.computercraft.client.gui.GuiComputer.*;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -104,21 +102,11 @@ public final class ItemPocketRenderer extends ItemMapLikeRenderer
|
|||||||
|
|
||||||
if( computer != null && terminal != null )
|
if( computer != null && terminal != null )
|
||||||
{
|
{
|
||||||
// If we've a computer and terminal then attempt to render it.
|
FixedWidthFontRenderer.drawTerminal( MARGIN, MARGIN, terminal, !computer.isColour(), MARGIN, MARGIN, MARGIN, MARGIN );
|
||||||
renderTerminal( terminal, !computer.isColour(), width, height );
|
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
// Otherwise render a plain background
|
FixedWidthFontRenderer.drawEmptyTerminal( 0, 0, width, height );
|
||||||
Minecraft.getMinecraft().getTextureManager().bindTexture( BACKGROUND );
|
|
||||||
|
|
||||||
Tessellator tessellator = Tessellator.getInstance();
|
|
||||||
BufferBuilder buffer = tessellator.getBuffer();
|
|
||||||
|
|
||||||
Colour black = Colour.Black;
|
|
||||||
buffer.begin( GL11.GL_QUADS, DefaultVertexFormats.POSITION );
|
|
||||||
renderTexture( buffer, 0, 0, 0, 0, width, height, black.getR(), black.getG(), black.getB() );
|
|
||||||
tessellator.draw();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
GlStateManager.enableDepth();
|
GlStateManager.enableDepth();
|
||||||
@@ -189,53 +177,6 @@ public final class ItemPocketRenderer extends ItemMapLikeRenderer
|
|||||||
GlStateManager.enableTexture2D();
|
GlStateManager.enableTexture2D();
|
||||||
}
|
}
|
||||||
|
|
||||||
private static void renderTerminal( Terminal terminal, boolean greyscale, int width, int height )
|
|
||||||
{
|
|
||||||
synchronized( terminal )
|
|
||||||
{
|
|
||||||
int termWidth = terminal.getWidth();
|
|
||||||
int termHeight = terminal.getHeight();
|
|
||||||
|
|
||||||
FixedWidthFontRenderer fontRenderer = FixedWidthFontRenderer.instance();
|
|
||||||
Palette palette = terminal.getPalette();
|
|
||||||
|
|
||||||
// Render top/bottom borders
|
|
||||||
TextBuffer emptyLine = new TextBuffer( ' ', termWidth );
|
|
||||||
fontRenderer.drawString(
|
|
||||||
emptyLine, MARGIN, 0,
|
|
||||||
terminal.getTextColourLine( 0 ), terminal.getBackgroundColourLine( 0 ), MARGIN, MARGIN, greyscale, palette
|
|
||||||
);
|
|
||||||
fontRenderer.drawString(
|
|
||||||
emptyLine, MARGIN, 2 * MARGIN + (termHeight - 1) * FixedWidthFontRenderer.FONT_HEIGHT,
|
|
||||||
terminal.getTextColourLine( termHeight - 1 ), terminal.getBackgroundColourLine( termHeight - 1 ), MARGIN, MARGIN, greyscale, palette
|
|
||||||
);
|
|
||||||
|
|
||||||
// Render the actual text
|
|
||||||
for( int line = 0; line < termWidth; line++ )
|
|
||||||
{
|
|
||||||
TextBuffer text = terminal.getLine( line );
|
|
||||||
TextBuffer colour = terminal.getTextColourLine( line );
|
|
||||||
TextBuffer backgroundColour = terminal.getBackgroundColourLine( line );
|
|
||||||
fontRenderer.drawString(
|
|
||||||
text, MARGIN, MARGIN + line * FONT_HEIGHT,
|
|
||||||
colour, backgroundColour, MARGIN, MARGIN, greyscale, palette
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
// And render the cursor;
|
|
||||||
int tx = terminal.getCursorX(), ty = terminal.getCursorY();
|
|
||||||
if( terminal.getCursorBlink() && FrameInfo.getGlobalCursorBlink() &&
|
|
||||||
tx >= 0 && ty >= 0 && tx < termWidth && ty < termHeight )
|
|
||||||
{
|
|
||||||
TextBuffer cursorColour = new TextBuffer( "0123456789abcdef".charAt( terminal.getTextColour() ), 1 );
|
|
||||||
fontRenderer.drawString(
|
|
||||||
new TextBuffer( '_', 1 ), MARGIN + FONT_WIDTH * tx, MARGIN + FONT_HEIGHT * ty,
|
|
||||||
cursorColour, null, 0, 0, greyscale, palette
|
|
||||||
);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
private static void renderTexture( BufferBuilder builder, int x, int y, int textureX, int textureY, int width, int height, float r, float g, float b )
|
private static void renderTexture( BufferBuilder builder, int x, int y, int textureX, int textureY, int width, int height, float r, float g, float b )
|
||||||
{
|
{
|
||||||
renderTexture( builder, x, y, textureX, textureY, width, height, width, height, r, g, b );
|
renderTexture( builder, x, y, textureX, textureY, width, height, width, height, r, g, b );
|
||||||
|
@@ -63,11 +63,12 @@ public final class PrintoutRenderer
|
|||||||
|
|
||||||
public static void drawText( int x, int y, int start, TextBuffer[] text, TextBuffer[] colours )
|
public static void drawText( int x, int y, int start, TextBuffer[] text, TextBuffer[] colours )
|
||||||
{
|
{
|
||||||
FixedWidthFontRenderer fontRenderer = FixedWidthFontRenderer.instance();
|
|
||||||
|
|
||||||
for( int line = 0; line < LINES_PER_PAGE && line < text.length; line++ )
|
for( int line = 0; line < LINES_PER_PAGE && line < text.length; line++ )
|
||||||
{
|
{
|
||||||
fontRenderer.drawString( text[start + line], x, y + line * FONT_HEIGHT, colours[start + line], null, 0, 0, false, Palette.DEFAULT );
|
FixedWidthFontRenderer.drawString(
|
||||||
|
x, y + line * FONT_HEIGHT, text[start + line], colours[start + line], null, Palette.DEFAULT,
|
||||||
|
false, 0, 0
|
||||||
|
);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -78,11 +79,13 @@ public final class PrintoutRenderer
|
|||||||
GlStateManager.enableTexture2D();
|
GlStateManager.enableTexture2D();
|
||||||
GlStateManager.tryBlendFuncSeparate( SourceFactor.SRC_ALPHA, DestFactor.ONE_MINUS_SRC_ALPHA, SourceFactor.ONE, DestFactor.ZERO );
|
GlStateManager.tryBlendFuncSeparate( SourceFactor.SRC_ALPHA, DestFactor.ONE_MINUS_SRC_ALPHA, SourceFactor.ONE, DestFactor.ZERO );
|
||||||
|
|
||||||
FixedWidthFontRenderer fontRenderer = FixedWidthFontRenderer.instance();
|
|
||||||
|
|
||||||
for( int line = 0; line < LINES_PER_PAGE && line < text.length; line++ )
|
for( int line = 0; line < LINES_PER_PAGE && line < text.length; line++ )
|
||||||
{
|
{
|
||||||
fontRenderer.drawString( new TextBuffer( text[start + line] ), x, y + line * FONT_HEIGHT, new TextBuffer( colours[start + line] ), null, 0, 0, false, Palette.DEFAULT );
|
FixedWidthFontRenderer.drawString(
|
||||||
|
x, y + line * FONT_HEIGHT,
|
||||||
|
new TextBuffer( text[start + line] ), new TextBuffer( colours[start + line] ),
|
||||||
|
null, Palette.DEFAULT, false, 0, 0
|
||||||
|
);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@@ -8,32 +8,33 @@ package dan200.computercraft.client.render;
|
|||||||
import dan200.computercraft.client.FrameInfo;
|
import dan200.computercraft.client.FrameInfo;
|
||||||
import dan200.computercraft.client.gui.FixedWidthFontRenderer;
|
import dan200.computercraft.client.gui.FixedWidthFontRenderer;
|
||||||
import dan200.computercraft.core.terminal.Terminal;
|
import dan200.computercraft.core.terminal.Terminal;
|
||||||
import dan200.computercraft.core.terminal.TextBuffer;
|
|
||||||
import dan200.computercraft.shared.peripheral.monitor.ClientMonitor;
|
import dan200.computercraft.shared.peripheral.monitor.ClientMonitor;
|
||||||
|
import dan200.computercraft.shared.peripheral.monitor.MonitorRenderer;
|
||||||
import dan200.computercraft.shared.peripheral.monitor.TileMonitor;
|
import dan200.computercraft.shared.peripheral.monitor.TileMonitor;
|
||||||
import dan200.computercraft.shared.util.Colour;
|
|
||||||
import dan200.computercraft.shared.util.DirectionUtil;
|
import dan200.computercraft.shared.util.DirectionUtil;
|
||||||
import dan200.computercraft.shared.util.Palette;
|
|
||||||
import net.minecraft.client.Minecraft;
|
import net.minecraft.client.Minecraft;
|
||||||
import net.minecraft.client.renderer.BufferBuilder;
|
import net.minecraft.client.renderer.BufferBuilder;
|
||||||
import net.minecraft.client.renderer.GlStateManager;
|
import net.minecraft.client.renderer.GlStateManager;
|
||||||
import net.minecraft.client.renderer.OpenGlHelper;
|
import net.minecraft.client.renderer.OpenGlHelper;
|
||||||
import net.minecraft.client.renderer.Tessellator;
|
import net.minecraft.client.renderer.Tessellator;
|
||||||
import net.minecraft.client.renderer.tileentity.TileEntitySpecialRenderer;
|
import net.minecraft.client.renderer.tileentity.TileEntitySpecialRenderer;
|
||||||
import net.minecraft.client.renderer.vertex.DefaultVertexFormats;
|
import net.minecraft.client.renderer.vertex.VertexBuffer;
|
||||||
import net.minecraft.util.EnumFacing;
|
import net.minecraft.util.EnumFacing;
|
||||||
import net.minecraft.util.math.BlockPos;
|
import net.minecraft.util.math.BlockPos;
|
||||||
import org.lwjgl.opengl.GL11;
|
import org.lwjgl.opengl.GL11;
|
||||||
|
|
||||||
|
import javax.annotation.Nonnull;
|
||||||
|
|
||||||
|
import static dan200.computercraft.shared.peripheral.monitor.TileMonitor.RENDER_MARGIN;
|
||||||
|
|
||||||
public class TileEntityMonitorRenderer extends TileEntitySpecialRenderer<TileMonitor>
|
public class TileEntityMonitorRenderer extends TileEntitySpecialRenderer<TileMonitor>
|
||||||
{
|
{
|
||||||
|
private static final float MARGIN = (float) (TileMonitor.RENDER_MARGIN * 1.1);
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public void render( TileMonitor tileEntity, double posX, double posY, double posZ, float f, int i, float f2 )
|
public void render( @Nonnull TileMonitor tileEntity, double posX, double posY, double posZ, float f, int i, float f2 )
|
||||||
{
|
{
|
||||||
if( tileEntity != null )
|
renderMonitorAt( tileEntity, posX, posY, posZ, f, i );
|
||||||
{
|
|
||||||
renderMonitorAt( tileEntity, posX, posY, posZ, f, i );
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
private static void renderMonitorAt( TileMonitor monitor, double posX, double posY, double posZ, float f, int i )
|
private static void renderMonitorAt( TileMonitor monitor, double posX, double posY, double posZ, float f, int i )
|
||||||
@@ -69,223 +70,141 @@ public class TileEntityMonitorRenderer extends TileEntitySpecialRenderer<TileMon
|
|||||||
float pitch = DirectionUtil.toPitchAngle( front );
|
float pitch = DirectionUtil.toPitchAngle( front );
|
||||||
|
|
||||||
GlStateManager.pushMatrix();
|
GlStateManager.pushMatrix();
|
||||||
try
|
|
||||||
|
// Setup initial transform
|
||||||
|
GlStateManager.translate( posX + 0.5, posY + 0.5, posZ + 0.5 );
|
||||||
|
GlStateManager.rotate( -yaw, 0.0f, 1.0f, 0.0f );
|
||||||
|
GlStateManager.rotate( pitch, 1.0f, 0.0f, 0.0f );
|
||||||
|
GlStateManager.translate(
|
||||||
|
-0.5 + TileMonitor.RENDER_BORDER + RENDER_MARGIN,
|
||||||
|
origin.getHeight() - 0.5 - (TileMonitor.RENDER_BORDER + RENDER_MARGIN),
|
||||||
|
0.5
|
||||||
|
);
|
||||||
|
double xSize = origin.getWidth() - 2.0 * (RENDER_MARGIN + TileMonitor.RENDER_BORDER);
|
||||||
|
double ySize = origin.getHeight() - 2.0 * (RENDER_MARGIN + TileMonitor.RENDER_BORDER);
|
||||||
|
|
||||||
|
// Get renderers
|
||||||
|
Minecraft mc = Minecraft.getMinecraft();
|
||||||
|
|
||||||
|
// Set up render state for monitors. We disable writing to the depth buffer (we draw a "blocker" later),
|
||||||
|
// and setup lighting so that we render with a glow.
|
||||||
|
GlStateManager.depthMask( false );
|
||||||
|
OpenGlHelper.setLightmapTextureCoords( OpenGlHelper.lightmapTexUnit, 0xFF, 0xFF );
|
||||||
|
GlStateManager.disableLighting();
|
||||||
|
mc.entityRenderer.disableLightmap();
|
||||||
|
|
||||||
|
Terminal terminal = originTerminal.getTerminal();
|
||||||
|
if( terminal != null )
|
||||||
{
|
{
|
||||||
// Setup initial transform
|
// Draw a terminal
|
||||||
GlStateManager.translate( posX + 0.5, posY + 0.5, posZ + 0.5 );
|
double xScale = xSize / (terminal.getWidth() * FixedWidthFontRenderer.FONT_WIDTH);
|
||||||
GlStateManager.rotate( -yaw, 0.0f, 1.0f, 0.0f );
|
double yScale = ySize / (terminal.getHeight() * FixedWidthFontRenderer.FONT_HEIGHT);
|
||||||
GlStateManager.rotate( pitch, 1.0f, 0.0f, 0.0f );
|
|
||||||
GlStateManager.translate(
|
|
||||||
-0.5 + TileMonitor.RENDER_BORDER + TileMonitor.RENDER_MARGIN,
|
|
||||||
origin.getHeight() - 0.5 - (TileMonitor.RENDER_BORDER + TileMonitor.RENDER_MARGIN),
|
|
||||||
0.5
|
|
||||||
);
|
|
||||||
double xSize = origin.getWidth() - 2.0 * (TileMonitor.RENDER_MARGIN + TileMonitor.RENDER_BORDER);
|
|
||||||
double ySize = origin.getHeight() - 2.0 * (TileMonitor.RENDER_MARGIN + TileMonitor.RENDER_BORDER);
|
|
||||||
|
|
||||||
// Get renderers
|
GlStateManager.pushMatrix();
|
||||||
Minecraft mc = Minecraft.getMinecraft();
|
GlStateManager.scale( (float) xScale, (float) -yScale, 1.0f );
|
||||||
Tessellator tessellator = Tessellator.getInstance();
|
|
||||||
BufferBuilder renderer = tessellator.getBuffer();
|
|
||||||
|
|
||||||
// Get terminal
|
renderTerminal( originTerminal, (float) (MARGIN / xScale), (float) (MARGIN / yScale) );
|
||||||
boolean redraw = originTerminal.pollTerminalChanged();
|
|
||||||
|
|
||||||
// Draw the contents
|
|
||||||
GlStateManager.depthMask( false );
|
|
||||||
OpenGlHelper.setLightmapTextureCoords( OpenGlHelper.lightmapTexUnit, 0xFF, 0xFF );
|
|
||||||
GlStateManager.disableLighting();
|
|
||||||
mc.entityRenderer.disableLightmap();
|
|
||||||
try
|
|
||||||
{
|
|
||||||
Terminal terminal = originTerminal.getTerminal();
|
|
||||||
if( terminal != null )
|
|
||||||
{
|
|
||||||
Palette palette = terminal.getPalette();
|
|
||||||
|
|
||||||
// Allocate display lists
|
|
||||||
if( originTerminal.renderDisplayLists == null )
|
|
||||||
{
|
|
||||||
originTerminal.createLists();
|
|
||||||
redraw = true;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Draw a terminal
|
|
||||||
boolean greyscale = !originTerminal.isColour();
|
|
||||||
int width = terminal.getWidth();
|
|
||||||
int height = terminal.getHeight();
|
|
||||||
int cursorX = terminal.getCursorX();
|
|
||||||
int cursorY = terminal.getCursorY();
|
|
||||||
FixedWidthFontRenderer fontRenderer = FixedWidthFontRenderer.instance();
|
|
||||||
|
|
||||||
GlStateManager.pushMatrix();
|
|
||||||
try
|
|
||||||
{
|
|
||||||
double xScale = xSize / (width * FixedWidthFontRenderer.FONT_WIDTH);
|
|
||||||
double yScale = ySize / (height * FixedWidthFontRenderer.FONT_HEIGHT);
|
|
||||||
GlStateManager.scale( xScale, -yScale, 1.0 );
|
|
||||||
|
|
||||||
// Draw background
|
|
||||||
mc.getTextureManager().bindTexture( FixedWidthFontRenderer.BACKGROUND );
|
|
||||||
if( redraw )
|
|
||||||
{
|
|
||||||
// Build background display list
|
|
||||||
GlStateManager.glNewList( originTerminal.renderDisplayLists[0], GL11.GL_COMPILE );
|
|
||||||
try
|
|
||||||
{
|
|
||||||
double marginXSize = TileMonitor.RENDER_MARGIN / xScale;
|
|
||||||
double marginYSize = TileMonitor.RENDER_MARGIN / yScale;
|
|
||||||
double marginSquash = marginYSize / FixedWidthFontRenderer.FONT_HEIGHT;
|
|
||||||
|
|
||||||
// Top and bottom margins
|
|
||||||
GlStateManager.pushMatrix();
|
|
||||||
try
|
|
||||||
{
|
|
||||||
GlStateManager.scale( 1.0, marginSquash, 1.0 );
|
|
||||||
GlStateManager.translate( 0.0, -marginYSize / marginSquash, 0.0 );
|
|
||||||
fontRenderer.drawStringBackgroundPart( 0, 0, terminal.getBackgroundColourLine( 0 ), marginXSize, marginXSize, greyscale, palette );
|
|
||||||
GlStateManager.translate( 0.0, (marginYSize + height * FixedWidthFontRenderer.FONT_HEIGHT) / marginSquash, 0.0 );
|
|
||||||
fontRenderer.drawStringBackgroundPart( 0, 0, terminal.getBackgroundColourLine( height - 1 ), marginXSize, marginXSize, greyscale, palette );
|
|
||||||
}
|
|
||||||
finally
|
|
||||||
{
|
|
||||||
GlStateManager.popMatrix();
|
|
||||||
}
|
|
||||||
|
|
||||||
// Backgrounds
|
|
||||||
for( int y = 0; y < height; y++ )
|
|
||||||
{
|
|
||||||
fontRenderer.drawStringBackgroundPart(
|
|
||||||
0, FixedWidthFontRenderer.FONT_HEIGHT * y,
|
|
||||||
terminal.getBackgroundColourLine( y ),
|
|
||||||
marginXSize, marginXSize,
|
|
||||||
greyscale,
|
|
||||||
palette
|
|
||||||
);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
finally
|
|
||||||
{
|
|
||||||
GlStateManager.glEndList();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
GlStateManager.callList( originTerminal.renderDisplayLists[0] );
|
|
||||||
GlStateManager.resetColor();
|
|
||||||
|
|
||||||
// Draw text
|
|
||||||
fontRenderer.bindFont();
|
|
||||||
if( redraw )
|
|
||||||
{
|
|
||||||
// Build text display list
|
|
||||||
GlStateManager.glNewList( originTerminal.renderDisplayLists[1], GL11.GL_COMPILE );
|
|
||||||
try
|
|
||||||
{
|
|
||||||
// Lines
|
|
||||||
for( int y = 0; y < height; y++ )
|
|
||||||
{
|
|
||||||
fontRenderer.drawStringTextPart(
|
|
||||||
0, FixedWidthFontRenderer.FONT_HEIGHT * y,
|
|
||||||
terminal.getLine( y ),
|
|
||||||
terminal.getTextColourLine( y ),
|
|
||||||
greyscale,
|
|
||||||
palette
|
|
||||||
);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
finally
|
|
||||||
{
|
|
||||||
GlStateManager.glEndList();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
GlStateManager.callList( originTerminal.renderDisplayLists[1] );
|
|
||||||
GlStateManager.resetColor();
|
|
||||||
|
|
||||||
// Draw cursor
|
|
||||||
fontRenderer.bindFont();
|
|
||||||
if( redraw )
|
|
||||||
{
|
|
||||||
// Build cursor display list
|
|
||||||
GlStateManager.glNewList( originTerminal.renderDisplayLists[2], GL11.GL_COMPILE );
|
|
||||||
try
|
|
||||||
{
|
|
||||||
// Cursor
|
|
||||||
if( terminal.getCursorBlink() && cursorX >= 0 && cursorX < width && cursorY >= 0 && cursorY < height )
|
|
||||||
{
|
|
||||||
TextBuffer cursor = new TextBuffer( "_" );
|
|
||||||
TextBuffer cursorColour = new TextBuffer( "0123456789abcdef".charAt( terminal.getTextColour() ), 1 );
|
|
||||||
fontRenderer.drawString(
|
|
||||||
cursor,
|
|
||||||
FixedWidthFontRenderer.FONT_WIDTH * cursorX,
|
|
||||||
FixedWidthFontRenderer.FONT_HEIGHT * cursorY,
|
|
||||||
cursorColour, null,
|
|
||||||
0, 0,
|
|
||||||
greyscale,
|
|
||||||
palette
|
|
||||||
);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
finally
|
|
||||||
{
|
|
||||||
GlStateManager.glEndList();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
if( FrameInfo.getGlobalCursorBlink() )
|
|
||||||
{
|
|
||||||
GlStateManager.callList( originTerminal.renderDisplayLists[2] );
|
|
||||||
GlStateManager.resetColor();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
finally
|
|
||||||
{
|
|
||||||
GlStateManager.popMatrix();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
// Draw a big black quad
|
|
||||||
mc.getTextureManager().bindTexture( FixedWidthFontRenderer.BACKGROUND );
|
|
||||||
final Colour colour = Colour.Black;
|
|
||||||
|
|
||||||
final float r = colour.getR();
|
|
||||||
final float g = colour.getG();
|
|
||||||
final float b = colour.getB();
|
|
||||||
|
|
||||||
renderer.begin( GL11.GL_TRIANGLE_STRIP, DefaultVertexFormats.POSITION_TEX_COLOR );
|
|
||||||
renderer.pos( -TileMonitor.RENDER_MARGIN, TileMonitor.RENDER_MARGIN, 0.0D ).tex( 0.0, 0.0 ).color( r, g, b, 1.0f ).endVertex();
|
|
||||||
renderer.pos( -TileMonitor.RENDER_MARGIN, -ySize - TileMonitor.RENDER_MARGIN, 0.0 ).tex( 0.0, 1.0 ).color( r, g, b, 1.0f ).endVertex();
|
|
||||||
renderer.pos( xSize + TileMonitor.RENDER_MARGIN, TileMonitor.RENDER_MARGIN, 0.0D ).tex( 1.0, 0.0 ).color( r, g, b, 1.0f ).endVertex();
|
|
||||||
renderer.pos( xSize + TileMonitor.RENDER_MARGIN, -ySize - TileMonitor.RENDER_MARGIN, 0.0 ).tex( 1.0, 1.0 ).color( r, g, b, 1.0f ).endVertex();
|
|
||||||
tessellator.draw();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
finally
|
|
||||||
{
|
|
||||||
GlStateManager.depthMask( true );
|
|
||||||
mc.entityRenderer.enableLightmap();
|
|
||||||
GlStateManager.enableLighting();
|
|
||||||
}
|
|
||||||
|
|
||||||
// Draw the depth blocker
|
|
||||||
GlStateManager.colorMask( false, false, false, false );
|
|
||||||
try
|
|
||||||
{
|
|
||||||
mc.getTextureManager().bindTexture( FixedWidthFontRenderer.BACKGROUND );
|
|
||||||
renderer.begin( GL11.GL_TRIANGLE_STRIP, DefaultVertexFormats.POSITION );
|
|
||||||
renderer.pos( -TileMonitor.RENDER_MARGIN, TileMonitor.RENDER_MARGIN, 0.0 ).endVertex();
|
|
||||||
renderer.pos( -TileMonitor.RENDER_MARGIN, -ySize - TileMonitor.RENDER_MARGIN, 0.0 ).endVertex();
|
|
||||||
renderer.pos( xSize + TileMonitor.RENDER_MARGIN, TileMonitor.RENDER_MARGIN, 0.0 ).endVertex();
|
|
||||||
renderer.pos( xSize + TileMonitor.RENDER_MARGIN, -ySize - TileMonitor.RENDER_MARGIN, 0.0 ).endVertex();
|
|
||||||
tessellator.draw();
|
|
||||||
}
|
|
||||||
finally
|
|
||||||
{
|
|
||||||
GlStateManager.colorMask( true, true, true, true );
|
|
||||||
}
|
|
||||||
}
|
|
||||||
finally
|
|
||||||
{
|
|
||||||
GlStateManager.color( 1.0f, 1.0f, 1.0f, 1.0f );
|
|
||||||
GlStateManager.popMatrix();
|
GlStateManager.popMatrix();
|
||||||
}
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
FixedWidthFontRenderer.drawEmptyTerminal(
|
||||||
|
-MARGIN, MARGIN,
|
||||||
|
(float) (xSize + 2 * MARGIN), (float) -(ySize + MARGIN * 2)
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Tear down render state for monitors.
|
||||||
|
GlStateManager.depthMask( true );
|
||||||
|
mc.entityRenderer.enableLightmap();
|
||||||
|
GlStateManager.enableLighting();
|
||||||
|
|
||||||
|
// Draw the depth blocker
|
||||||
|
GlStateManager.colorMask( false, false, false, false );
|
||||||
|
FixedWidthFontRenderer.drawBlocker(
|
||||||
|
(float) -TileMonitor.RENDER_MARGIN, (float) TileMonitor.RENDER_MARGIN,
|
||||||
|
(float) (xSize + 2 * TileMonitor.RENDER_MARGIN), (float) -(ySize + TileMonitor.RENDER_MARGIN * 2)
|
||||||
|
);
|
||||||
|
GlStateManager.colorMask( true, true, true, true );
|
||||||
|
|
||||||
|
GlStateManager.popMatrix();
|
||||||
|
}
|
||||||
|
|
||||||
|
private static void renderTerminal( ClientMonitor monitor, float xMargin, float yMargin )
|
||||||
|
{
|
||||||
|
Tessellator tessellator = Tessellator.getInstance();
|
||||||
|
BufferBuilder buffer = tessellator.getBuffer();
|
||||||
|
|
||||||
|
boolean redraw = monitor.pollTerminalChanged();
|
||||||
|
|
||||||
|
// Setup the buffers if needed. We get the renderer here, to avoid the (unlikely) race condition between
|
||||||
|
// creating the buffers and rendering.
|
||||||
|
MonitorRenderer renderer = MonitorRenderer.current();
|
||||||
|
if( monitor.createBuffer( renderer ) ) redraw = true;
|
||||||
|
|
||||||
|
FixedWidthFontRenderer.bindFont();
|
||||||
|
|
||||||
|
switch( renderer )
|
||||||
|
{
|
||||||
|
case VBO:
|
||||||
|
{
|
||||||
|
VertexBuffer vbo = monitor.buffer;
|
||||||
|
if( redraw )
|
||||||
|
{
|
||||||
|
renderTerminalTo( monitor, buffer, xMargin, yMargin );
|
||||||
|
buffer.finishDrawing();
|
||||||
|
buffer.reset();
|
||||||
|
vbo.bufferData( buffer.getByteBuffer() );
|
||||||
|
}
|
||||||
|
|
||||||
|
vbo.bindBuffer();
|
||||||
|
setupBufferFormat();
|
||||||
|
vbo.drawArrays( GL11.GL_TRIANGLES );
|
||||||
|
vbo.unbindBuffer();
|
||||||
|
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
case DISPLAY_LIST:
|
||||||
|
if( redraw )
|
||||||
|
{
|
||||||
|
GlStateManager.glNewList( monitor.displayList, GL11.GL_COMPILE );
|
||||||
|
renderTerminalTo( monitor, buffer, xMargin, yMargin );
|
||||||
|
tessellator.draw();
|
||||||
|
GlStateManager.glEndList();
|
||||||
|
}
|
||||||
|
|
||||||
|
GlStateManager.callList( monitor.displayList );
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
// We don't draw the cursor with a buffer, as it's dynamic and so we'll end up refreshing far more than is
|
||||||
|
// reasonable.
|
||||||
|
FixedWidthFontRenderer.begin( buffer );
|
||||||
|
FixedWidthFontRenderer.drawCursor( buffer, 0, 0, monitor.getTerminal(), !monitor.isColour() );
|
||||||
|
tessellator.draw();
|
||||||
|
}
|
||||||
|
|
||||||
|
private static void renderTerminalTo( ClientMonitor monitor, BufferBuilder buffer, float xMargin, float yMargin )
|
||||||
|
{
|
||||||
|
FixedWidthFontRenderer.begin( buffer );
|
||||||
|
FixedWidthFontRenderer.drawTerminalWithoutCursor(
|
||||||
|
buffer, 0, 0,
|
||||||
|
monitor.getTerminal(), !monitor.isColour(), yMargin, yMargin, xMargin, xMargin
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
public static void setupBufferFormat()
|
||||||
|
{
|
||||||
|
int stride = FixedWidthFontRenderer.POSITION_COLOR_TEX.getSize();
|
||||||
|
GlStateManager.glVertexPointer( 3, GL11.GL_FLOAT, stride, 0 );
|
||||||
|
GlStateManager.glEnableClientState( GL11.GL_VERTEX_ARRAY );
|
||||||
|
|
||||||
|
GlStateManager.glColorPointer( 4, GL11.GL_UNSIGNED_BYTE, stride, 12 );
|
||||||
|
GlStateManager.glEnableClientState( GL11.GL_COLOR_ARRAY );
|
||||||
|
|
||||||
|
GlStateManager.glTexCoordPointer( 2, GL11.GL_FLOAT, stride, 16 );
|
||||||
|
GlStateManager.glEnableClientState( GL11.GL_TEXTURE_COORD_ARRAY );
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@@ -11,6 +11,7 @@ import dan200.computercraft.ComputerCraft;
|
|||||||
import dan200.computercraft.api.turtle.event.TurtleAction;
|
import dan200.computercraft.api.turtle.event.TurtleAction;
|
||||||
import dan200.computercraft.core.apis.AddressPredicate;
|
import dan200.computercraft.core.apis.AddressPredicate;
|
||||||
import dan200.computercraft.core.apis.http.websocket.Websocket;
|
import dan200.computercraft.core.apis.http.websocket.Websocket;
|
||||||
|
import dan200.computercraft.shared.peripheral.monitor.MonitorRenderer;
|
||||||
import net.minecraftforge.common.config.ConfigCategory;
|
import net.minecraftforge.common.config.ConfigCategory;
|
||||||
import net.minecraftforge.common.config.ConfigElement;
|
import net.minecraftforge.common.config.ConfigElement;
|
||||||
import net.minecraftforge.common.config.Configuration;
|
import net.minecraftforge.common.config.Configuration;
|
||||||
@@ -69,6 +70,7 @@ public final class Config
|
|||||||
private static Property modemRangeDuringStorm;
|
private static Property modemRangeDuringStorm;
|
||||||
private static Property modemHighAltitudeRangeDuringStorm;
|
private static Property modemHighAltitudeRangeDuringStorm;
|
||||||
private static Property maxNotesPerTick;
|
private static Property maxNotesPerTick;
|
||||||
|
private static Property monitorRenderer;
|
||||||
|
|
||||||
private static Property turtlesNeedFuel;
|
private static Property turtlesNeedFuel;
|
||||||
private static Property turtleFuelLimit;
|
private static Property turtleFuelLimit;
|
||||||
@@ -264,9 +266,15 @@ public final class Config
|
|||||||
maxNotesPerTick.setComment( "Maximum amount of notes a speaker can play at once" );
|
maxNotesPerTick.setComment( "Maximum amount of notes a speaker can play at once" );
|
||||||
maxNotesPerTick.setMinValue( 1 );
|
maxNotesPerTick.setMinValue( 1 );
|
||||||
|
|
||||||
|
monitorRenderer = config.get( CATEGORY_PERIPHERAL, "monitor_renderer", ComputerCraft.monitorRenderer.displayName() );
|
||||||
|
monitorRenderer.setComment( "The renderer to use for monitors. Generally this should be kept at \"best\" - if " +
|
||||||
|
"monitors have performance issues, you may wish to experiment with alternative renderers." );
|
||||||
|
monitorRenderer.setValidValues( MonitorRenderer.NAMES );
|
||||||
|
|
||||||
setOrder(
|
setOrder(
|
||||||
CATEGORY_PERIPHERAL,
|
CATEGORY_PERIPHERAL,
|
||||||
commandBlockEnabled, modemRange, modemHighAltitudeRange, modemRangeDuringStorm, modemHighAltitudeRangeDuringStorm, maxNotesPerTick
|
commandBlockEnabled, modemRange, modemHighAltitudeRange, modemRangeDuringStorm, modemHighAltitudeRangeDuringStorm, maxNotesPerTick,
|
||||||
|
monitorRenderer
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -459,6 +467,7 @@ public final class Config
|
|||||||
ComputerCraft.modem_highAltitudeRange = Math.min( modemHighAltitudeRange.getInt(), MODEM_MAX_RANGE );
|
ComputerCraft.modem_highAltitudeRange = Math.min( modemHighAltitudeRange.getInt(), MODEM_MAX_RANGE );
|
||||||
ComputerCraft.modem_rangeDuringStorm = Math.min( modemRangeDuringStorm.getInt(), MODEM_MAX_RANGE );
|
ComputerCraft.modem_rangeDuringStorm = Math.min( modemRangeDuringStorm.getInt(), MODEM_MAX_RANGE );
|
||||||
ComputerCraft.modem_highAltitudeRangeDuringStorm = Math.min( modemHighAltitudeRangeDuringStorm.getInt(), MODEM_MAX_RANGE );
|
ComputerCraft.modem_highAltitudeRangeDuringStorm = Math.min( modemHighAltitudeRangeDuringStorm.getInt(), MODEM_MAX_RANGE );
|
||||||
|
ComputerCraft.monitorRenderer = MonitorRenderer.ofString( monitorRenderer.getString() );
|
||||||
|
|
||||||
// Turtles
|
// Turtles
|
||||||
ComputerCraft.turtlesNeedFuel = turtlesNeedFuel.getBoolean();
|
ComputerCraft.turtlesNeedFuel = turtlesNeedFuel.getBoolean();
|
||||||
|
@@ -5,8 +5,10 @@
|
|||||||
*/
|
*/
|
||||||
package dan200.computercraft.shared.peripheral.monitor;
|
package dan200.computercraft.shared.peripheral.monitor;
|
||||||
|
|
||||||
|
import dan200.computercraft.client.gui.FixedWidthFontRenderer;
|
||||||
import dan200.computercraft.shared.common.ClientTerminal;
|
import dan200.computercraft.shared.common.ClientTerminal;
|
||||||
import net.minecraft.client.renderer.GlStateManager;
|
import net.minecraft.client.renderer.GLAllocation;
|
||||||
|
import net.minecraft.client.renderer.vertex.VertexBuffer;
|
||||||
import net.minecraft.util.math.BlockPos;
|
import net.minecraft.util.math.BlockPos;
|
||||||
import net.minecraftforge.fml.relauncher.Side;
|
import net.minecraftforge.fml.relauncher.Side;
|
||||||
import net.minecraftforge.fml.relauncher.SideOnly;
|
import net.minecraftforge.fml.relauncher.SideOnly;
|
||||||
@@ -15,7 +17,7 @@ import java.util.HashSet;
|
|||||||
import java.util.Iterator;
|
import java.util.Iterator;
|
||||||
import java.util.Set;
|
import java.util.Set;
|
||||||
|
|
||||||
public class ClientMonitor extends ClientTerminal
|
public final class ClientMonitor extends ClientTerminal
|
||||||
{
|
{
|
||||||
private static final Set<ClientMonitor> allMonitors = new HashSet<>();
|
private static final Set<ClientMonitor> allMonitors = new HashSet<>();
|
||||||
|
|
||||||
@@ -23,7 +25,9 @@ public class ClientMonitor extends ClientTerminal
|
|||||||
|
|
||||||
public long lastRenderFrame = -1;
|
public long lastRenderFrame = -1;
|
||||||
public BlockPos lastRenderPos = null;
|
public BlockPos lastRenderPos = null;
|
||||||
public int[] renderDisplayLists = null;
|
|
||||||
|
public VertexBuffer buffer;
|
||||||
|
public int displayList = 0;
|
||||||
|
|
||||||
public ClientMonitor( boolean colour, TileMonitor origin )
|
public ClientMonitor( boolean colour, TileMonitor origin )
|
||||||
{
|
{
|
||||||
@@ -36,41 +40,72 @@ public class ClientMonitor extends ClientTerminal
|
|||||||
return origin;
|
return origin;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Create the appropriate buffer if needed.
|
||||||
|
*
|
||||||
|
* @param renderer The renderer to use. This can be fetched from {@link #renderer()}.
|
||||||
|
* @return If a buffer was created. This will return {@code false} if we already have an appropriate buffer,
|
||||||
|
* or this mode does not require one.
|
||||||
|
*/
|
||||||
@SideOnly( Side.CLIENT )
|
@SideOnly( Side.CLIENT )
|
||||||
public void createLists()
|
public boolean createBuffer( MonitorRenderer renderer )
|
||||||
{
|
{
|
||||||
if( renderDisplayLists == null )
|
switch( renderer )
|
||||||
{
|
{
|
||||||
renderDisplayLists = new int[3];
|
case VBO:
|
||||||
|
if( buffer != null ) return false;
|
||||||
|
|
||||||
for( int i = 0; i < renderDisplayLists.length; i++ )
|
deleteBuffers();
|
||||||
{
|
buffer = new VertexBuffer( FixedWidthFontRenderer.POSITION_COLOR_TEX );
|
||||||
renderDisplayLists[i] = GlStateManager.glGenLists( 1 );
|
addMonitor();
|
||||||
}
|
return true;
|
||||||
|
case DISPLAY_LIST:
|
||||||
|
if( displayList != 0 ) return false;
|
||||||
|
|
||||||
synchronized( allMonitors )
|
deleteBuffers();
|
||||||
{
|
displayList = GLAllocation.generateDisplayLists( 1 );
|
||||||
allMonitors.add( this );
|
addMonitor();
|
||||||
}
|
return true;
|
||||||
|
|
||||||
|
default:
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private void addMonitor()
|
||||||
|
{
|
||||||
|
synchronized( allMonitors )
|
||||||
|
{
|
||||||
|
allMonitors.add( this );
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private void deleteBuffers()
|
||||||
|
{
|
||||||
|
if( buffer != null )
|
||||||
|
{
|
||||||
|
buffer.deleteGlBuffers();
|
||||||
|
buffer = null;
|
||||||
|
}
|
||||||
|
|
||||||
|
if( displayList != 0 )
|
||||||
|
{
|
||||||
|
GLAllocation.deleteDisplayLists( displayList );
|
||||||
|
displayList = 0;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@SideOnly( Side.CLIENT )
|
@SideOnly( Side.CLIENT )
|
||||||
public void destroy()
|
public void destroy()
|
||||||
{
|
{
|
||||||
if( renderDisplayLists != null )
|
if( buffer != null || displayList != 0 )
|
||||||
{
|
{
|
||||||
synchronized( allMonitors )
|
synchronized( allMonitors )
|
||||||
{
|
{
|
||||||
allMonitors.remove( this );
|
allMonitors.remove( this );
|
||||||
}
|
}
|
||||||
|
|
||||||
for( int list : renderDisplayLists )
|
deleteBuffers();
|
||||||
{
|
|
||||||
GlStateManager.glDeleteLists( list, 1 );
|
|
||||||
}
|
|
||||||
|
|
||||||
renderDisplayLists = null;
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -82,14 +117,7 @@ public class ClientMonitor extends ClientTerminal
|
|||||||
for( Iterator<ClientMonitor> iterator = allMonitors.iterator(); iterator.hasNext(); )
|
for( Iterator<ClientMonitor> iterator = allMonitors.iterator(); iterator.hasNext(); )
|
||||||
{
|
{
|
||||||
ClientMonitor monitor = iterator.next();
|
ClientMonitor monitor = iterator.next();
|
||||||
if( monitor.renderDisplayLists != null )
|
monitor.deleteBuffers();
|
||||||
{
|
|
||||||
for( int list : monitor.renderDisplayLists )
|
|
||||||
{
|
|
||||||
GlStateManager.glDeleteLists( list, 1 );
|
|
||||||
}
|
|
||||||
monitor.renderDisplayLists = null;
|
|
||||||
}
|
|
||||||
|
|
||||||
iterator.remove();
|
iterator.remove();
|
||||||
}
|
}
|
||||||
|
@@ -0,0 +1,105 @@
|
|||||||
|
/*
|
||||||
|
* This file is part of ComputerCraft - http://www.computercraft.info
|
||||||
|
* Copyright Daniel Ratcliffe, 2011-2020. Do not distribute without permission.
|
||||||
|
* Send enquiries to dratcliffe@gmail.com
|
||||||
|
*/
|
||||||
|
|
||||||
|
package dan200.computercraft.shared.peripheral.monitor;
|
||||||
|
|
||||||
|
import dan200.computercraft.ComputerCraft;
|
||||||
|
import dan200.computercraft.client.render.TileEntityMonitorRenderer;
|
||||||
|
import net.minecraft.client.renderer.OpenGlHelper;
|
||||||
|
|
||||||
|
import javax.annotation.Nonnull;
|
||||||
|
import java.util.Locale;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* The render type to use for monitors.
|
||||||
|
*
|
||||||
|
* @see TileEntityMonitorRenderer
|
||||||
|
* @see ClientMonitor
|
||||||
|
*/
|
||||||
|
public enum MonitorRenderer
|
||||||
|
{
|
||||||
|
/**
|
||||||
|
* Determine the best monitor backend.
|
||||||
|
*/
|
||||||
|
BEST,
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Render using VBOs. This is the default when supported.
|
||||||
|
*
|
||||||
|
* @see net.minecraft.client.renderer.vertex.VertexBuffer
|
||||||
|
*/
|
||||||
|
VBO,
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Render using display lists.
|
||||||
|
*
|
||||||
|
* @see net.minecraft.client.renderer.GLAllocation#generateDisplayLists(int)
|
||||||
|
*/
|
||||||
|
DISPLAY_LIST;
|
||||||
|
|
||||||
|
private static final MonitorRenderer[] VALUES = values();
|
||||||
|
public static final String[] NAMES;
|
||||||
|
|
||||||
|
private final String displayName = "gui.computercraft:config.peripheral.monitor_renderer." + name().toLowerCase( Locale.ROOT );
|
||||||
|
|
||||||
|
static
|
||||||
|
{
|
||||||
|
NAMES = new String[VALUES.length];
|
||||||
|
for( int i = 0; i < VALUES.length; i++ ) NAMES[i] = VALUES[i].displayName();
|
||||||
|
}
|
||||||
|
|
||||||
|
public String displayName()
|
||||||
|
{
|
||||||
|
return displayName;
|
||||||
|
}
|
||||||
|
|
||||||
|
@Nonnull
|
||||||
|
public static MonitorRenderer ofString( String name )
|
||||||
|
{
|
||||||
|
for( MonitorRenderer backend : VALUES )
|
||||||
|
{
|
||||||
|
if( backend.displayName.equalsIgnoreCase( name ) || backend.name().equalsIgnoreCase( name ) )
|
||||||
|
{
|
||||||
|
return backend;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
ComputerCraft.log.warn( "Unknown monitor renderer {}. Falling back to default.", name );
|
||||||
|
return BEST;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Get the current renderer to use.
|
||||||
|
*
|
||||||
|
* @return The current renderer. Will not return {@link MonitorRenderer#BEST}.
|
||||||
|
*/
|
||||||
|
@Nonnull
|
||||||
|
public static MonitorRenderer current()
|
||||||
|
{
|
||||||
|
MonitorRenderer current = ComputerCraft.monitorRenderer;
|
||||||
|
switch( current )
|
||||||
|
{
|
||||||
|
case BEST:
|
||||||
|
return best();
|
||||||
|
case VBO:
|
||||||
|
if( !OpenGlHelper.vboSupported )
|
||||||
|
{
|
||||||
|
ComputerCraft.log.warn( "VBOs are not supported on your graphics card. Falling back to default." );
|
||||||
|
ComputerCraft.monitorRenderer = BEST;
|
||||||
|
return best();
|
||||||
|
}
|
||||||
|
|
||||||
|
return VBO;
|
||||||
|
default:
|
||||||
|
return current;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private static MonitorRenderer best()
|
||||||
|
{
|
||||||
|
return OpenGlHelper.vboSupported ? VBO : DISPLAY_LIST;
|
||||||
|
}
|
||||||
|
}
|
@@ -48,13 +48,13 @@ public class Palette
|
|||||||
{
|
{
|
||||||
if( i >= 0 && i < colours.length )
|
if( i >= 0 && i < colours.length )
|
||||||
{
|
{
|
||||||
setColour( i, Colour.values()[i] );
|
setColour( i, Colour.VALUES[i] );
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public void resetColours()
|
public void resetColours()
|
||||||
{
|
{
|
||||||
for( int i = 0; i < Colour.values().length; i++ )
|
for( int i = 0; i < Colour.VALUES.length; i++ )
|
||||||
{
|
{
|
||||||
resetColour( i );
|
resetColour( i );
|
||||||
}
|
}
|
||||||
|
@@ -185,6 +185,10 @@ gui.computercraft:config.peripheral.modem_high_altitude_range=Modem range (high-
|
|||||||
gui.computercraft:config.peripheral.modem_range_during_storm=Modem range (bad weather)
|
gui.computercraft:config.peripheral.modem_range_during_storm=Modem range (bad weather)
|
||||||
gui.computercraft:config.peripheral.modem_high_altitude_range_during_storm=Modem range (high-altitude, bad weather)
|
gui.computercraft:config.peripheral.modem_high_altitude_range_during_storm=Modem range (high-altitude, bad weather)
|
||||||
gui.computercraft:config.peripheral.max_notes_per_tick=Maximum notes that a computer can play at once
|
gui.computercraft:config.peripheral.max_notes_per_tick=Maximum notes that a computer can play at once
|
||||||
|
gui.computercraft:config.peripheral.monitor_renderer=Monitor renderer
|
||||||
|
gui.computercraft:config.peripheral.monitor_renderer.best=Best
|
||||||
|
gui.computercraft:config.peripheral.monitor_renderer.vbo=Vertex Buffers
|
||||||
|
gui.computercraft:config.peripheral.monitor_renderer.display_list=Display Lists
|
||||||
|
|
||||||
gui.computercraft:config.turtle=Turtles
|
gui.computercraft:config.turtle=Turtles
|
||||||
gui.computercraft:config.turtle.need_fuel=Enable fuel
|
gui.computercraft:config.turtle.need_fuel=Enable fuel
|
||||||
|
Binary file not shown.
Before Width: | Height: | Size: 123 B |
Binary file not shown.
Before Width: | Height: | Size: 1.2 KiB After Width: | Height: | Size: 3.8 KiB |
Reference in New Issue
Block a user