/* * This file is part of Applied Energistics 2. * Copyright (c) 2013 - 2014, AlgorithmX2, All rights reserved. * * Applied Energistics 2 is free software: you can redistribute it and/or modify * it under the terms of the GNU Lesser General Public License as published by * the Free Software Foundation, either version 3 of the License, or * (at your option) any later version. * * Applied Energistics 2 is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU Lesser General Public License for more details. * * You should have received a copy of the GNU Lesser General Public License * along with Applied Energistics 2. If not, see . */ package appeng.client.gui.implementations; import java.io.IOException; import java.util.ArrayList; import java.util.Iterator; import java.util.LinkedList; import java.util.List; import java.util.concurrent.TimeUnit; import com.google.common.base.Joiner; import org.apache.commons.lang3.time.DurationFormatUtils; import org.lwjgl.opengl.GL11; import net.minecraft.client.gui.GuiButton; import net.minecraft.entity.player.InventoryPlayer; import net.minecraft.item.ItemStack; import appeng.api.AEApi; import appeng.api.config.SortDir; import appeng.api.config.SortOrder; import appeng.api.config.ViewItems; import appeng.api.storage.data.IAEItemStack; import appeng.api.storage.data.IItemList; import appeng.api.util.AEColor; import appeng.client.gui.AEBaseGui; import appeng.client.gui.widgets.GuiScrollbar; import appeng.client.gui.widgets.ISortSource; import appeng.container.implementations.ContainerCraftingCPU; import appeng.core.AEConfig; import appeng.core.AELog; import appeng.core.localization.GuiText; import appeng.core.sync.network.NetworkHandler; import appeng.core.sync.packets.PacketValueConfig; import appeng.util.Platform; import appeng.util.ReadableNumberConverter; public class GuiCraftingCPU extends AEBaseGui implements ISortSource { private final static int GUI_HEIGHT = 184; private final static int GUI_WIDTH = 238; private final static int DISPLAYED_ROWS = 6; private final static int TEXT_COLOR = 0x404040; private final static int BACKGROUND_ALPHA = 0x5A000000; private final static int SECTION_LENGTH = 67; private final static int SCROLLBAR_TOP = 19; private final static int SCROLLBAR_LEFT = 218; private final static int SCROLLBAR_HEIGHT = 137; private final static int CANCEL_LEFT_OFFSET = 163; private final static int CANCEL_TOP_OFFSET = 25; private final static int CANCEL_HEIGHT = 20; private final static int CANCEL_WIDTH = 50; private final static int ETA_TOP_OFFSET = 165; private final static int ETA_LEFT_OFFSET = 8; private final static int ITEMSTACK_LEFT_OFFSET = 9; private final static int ITEMSTACK_TOP_OFFSET = 22; private final ContainerCraftingCPU craftingCpu; private IItemList storage = AEApi.instance().storage().createItemList(); private IItemList active = AEApi.instance().storage().createItemList(); private IItemList pending = AEApi.instance().storage().createItemList(); private List visual = new ArrayList(); private GuiButton cancel; private int tooltip = -1; public GuiCraftingCPU( InventoryPlayer inventoryPlayer, Object te ) { this( new ContainerCraftingCPU( inventoryPlayer, te ) ); } protected GuiCraftingCPU( ContainerCraftingCPU container ) { super( container ); this.craftingCpu = container; this.ySize = GUI_HEIGHT; this.xSize = GUI_WIDTH; this.myScrollBar = new GuiScrollbar(); } public void clearItems() { this.storage = AEApi.instance().storage().createItemList(); this.active = AEApi.instance().storage().createItemList(); this.pending = AEApi.instance().storage().createItemList(); this.visual = new ArrayList(); } @Override protected void actionPerformed( GuiButton btn ) { super.actionPerformed( btn ); if( this.cancel == btn ) { try { NetworkHandler.instance.sendToServer( new PacketValueConfig( "TileCrafting.Cancel", "Cancel" ) ); } catch( IOException e ) { AELog.error( e ); } } } @Override public void initGui() { super.initGui(); this.setScrollBar(); this.cancel = new GuiButton( 0, this.guiLeft + CANCEL_LEFT_OFFSET, this.guiTop + this.ySize - CANCEL_TOP_OFFSET, CANCEL_WIDTH, CANCEL_HEIGHT, GuiText.Cancel.getLocal() ); this.buttonList.add( this.cancel ); } private void setScrollBar() { int size = this.visual.size(); this.myScrollBar.setTop( SCROLLBAR_TOP ).setLeft( SCROLLBAR_LEFT ).setHeight( SCROLLBAR_HEIGHT ); this.myScrollBar.setRange( 0, ( size + 2 ) / 3 - DISPLAYED_ROWS, 1 ); } @Override public void drawScreen( int mouseX, int mouseY, float btn ) { this.cancel.enabled = !this.visual.isEmpty(); int x = 0; int y = 0; final int gx = ( this.width - this.xSize ) / 2; final int gy = ( this.height - this.ySize ) / 2; final int offY = 23; this.tooltip = -1; for( int z = 0; z <= 4 * 5; z++ ) { int minX = gx + 9 + x * 67; int minY = gy + 22 + y * offY; if( minX < mouseX && minX + 67 > mouseX ) { if( minY < mouseY && minY + offY - 2 > mouseY ) { this.tooltip = z; break; } } x++; if( x > 2 ) { y++; x = 0; } } super.drawScreen( mouseX, mouseY, btn ); } @Override public void drawFG( int offsetX, int offsetY, int mouseX, int mouseY ) { ReadableNumberConverter converter = ReadableNumberConverter.INSTANCE; this.fontRendererObj.drawString( this.getGuiDisplayName( GuiText.CraftingStatus.getLocal() ), 8, 7, TEXT_COLOR ); if( this.craftingCpu.eta > 0 ) { final String etaText = this.getGuiDisplayName( GuiText.ETA.getLocal() ); final long etaInMilliseconds = TimeUnit.MILLISECONDS.convert( this.craftingCpu.eta, TimeUnit.NANOSECONDS ); final String etaTimeText = DurationFormatUtils.formatDuration( etaInMilliseconds, this.getGuiDisplayName( GuiText.ETAFormat.getLocal() ) ); this.fontRendererObj.drawString( String.format( etaText, etaTimeText ), ETA_LEFT_OFFSET, ETA_TOP_OFFSET, TEXT_COLOR ); } int x = 0; int y = 0; final int viewStart = this.myScrollBar.getCurrentScroll() * 3; final int viewEnd = viewStart + 3 * 6; String dspToolTip = ""; List lineList = new LinkedList(); int toolPosX = 0; int toolPosY = 0; int offY = 23; for( int z = viewStart; z < Math.min( viewEnd, this.visual.size() ); z++ ) { IAEItemStack refStack = this.visual.get( z );// repo.getReferenceItem( z ); if( refStack != null ) { GL11.glPushMatrix(); GL11.glScaled( 0.5, 0.5, 0.5 ); final IAEItemStack stored = this.storage.findPrecise( refStack ); final IAEItemStack activeStack = this.active.findPrecise( refStack ); final IAEItemStack pendingStack = this.pending.findPrecise( refStack ); int lines = 0; boolean active = false; boolean scheduled = false; if( stored != null && stored.getStackSize() > 0 ) { lines++; } if( activeStack != null && activeStack.getStackSize() > 0 ) { lines++; active = true; } if( pendingStack != null && pendingStack.getStackSize() > 0 ) { lines++; scheduled = true; } if( AEConfig.instance.useColoredCraftingStatus && ( active || scheduled ) ) { int bgColor = ( active ? AEColor.Green.blackVariant : AEColor.Yellow.blackVariant ) | BACKGROUND_ALPHA; int startX = ( x * ( 1 + SECTION_LENGTH ) + ITEMSTACK_LEFT_OFFSET ) * 2; int startY = ( ( y * offY + ITEMSTACK_TOP_OFFSET ) - 3 ) * 2; drawRect( startX, startY, startX + ( SECTION_LENGTH * 2 ), startY + ( offY * 2 ) - 2, bgColor ); } int negY = ( ( lines - 1 ) * 5 ) / 2; int downY = 0; if( stored != null && stored.getStackSize() > 0 ) { final String str = GuiText.Stored.getLocal() + ": " + converter.toWideReadableForm( stored.getStackSize() ); int w = 4 + this.fontRendererObj.getStringWidth( str ); this.fontRendererObj.drawString( str, (int) ( ( x * ( 1 + SECTION_LENGTH ) + ITEMSTACK_LEFT_OFFSET + SECTION_LENGTH - 19 - ( w * 0.5 ) ) * 2 ), ( y * offY + ITEMSTACK_TOP_OFFSET + 6 - negY + downY ) * 2, TEXT_COLOR ); if( this.tooltip == z - viewStart ) { lineList.add( GuiText.Stored.getLocal() + ": " + Long.toString( stored.getStackSize() ) ); } downY += 5; } if( activeStack != null && activeStack.getStackSize() > 0 ) { final String str = GuiText.Crafting.getLocal() + ": " + converter.toWideReadableForm( activeStack.getStackSize() ); final int w = 4 + this.fontRendererObj.getStringWidth( str ); this.fontRendererObj.drawString( str, (int) ( ( x * ( 1 + SECTION_LENGTH ) + ITEMSTACK_LEFT_OFFSET + SECTION_LENGTH - 19 - ( w * 0.5 ) ) * 2 ), ( y * offY + ITEMSTACK_TOP_OFFSET + 6 - negY + downY ) * 2, TEXT_COLOR ); if( this.tooltip == z - viewStart ) { lineList.add( GuiText.Crafting.getLocal() + ": " + Long.toString( activeStack.getStackSize() ) ); } downY += 5; } if( pendingStack != null && pendingStack.getStackSize() > 0 ) { final String str = GuiText.Scheduled.getLocal() + ": " + converter.toWideReadableForm( pendingStack.getStackSize() ); final int w = 4 + this.fontRendererObj.getStringWidth( str ); this.fontRendererObj.drawString( str, (int) ( ( x * ( 1 + SECTION_LENGTH ) + ITEMSTACK_LEFT_OFFSET + SECTION_LENGTH - 19 - ( w * 0.5 ) ) * 2 ), ( y * offY + ITEMSTACK_TOP_OFFSET + 6 - negY + downY ) * 2, TEXT_COLOR ); if( this.tooltip == z - viewStart ) { lineList.add( GuiText.Scheduled.getLocal() + ": " + Long.toString( pendingStack.getStackSize() ) ); } } GL11.glPopMatrix(); int posX = x * ( 1 + SECTION_LENGTH ) + ITEMSTACK_LEFT_OFFSET + SECTION_LENGTH - 19; int posY = y * offY + ITEMSTACK_TOP_OFFSET; ItemStack is = refStack.copy().getItemStack(); if( this.tooltip == z - viewStart ) { dspToolTip = Platform.getItemDisplayName( is ); if( lineList.size() > 0 ) { dspToolTip = dspToolTip + '\n' + Joiner.on( "\n" ).join( lineList ); } toolPosX = x * ( 1 + SECTION_LENGTH ) + ITEMSTACK_LEFT_OFFSET + SECTION_LENGTH - 8; toolPosY = y * offY + ITEMSTACK_TOP_OFFSET; } this.drawItem( posX, posY, is ); x++; if( x > 2 ) { y++; x = 0; } } } if( this.tooltip >= 0 && dspToolTip.length() > 0 ) { GL11.glPushAttrib( GL11.GL_ALL_ATTRIB_BITS ); this.drawTooltip( toolPosX, toolPosY + 10, 0, dspToolTip ); GL11.glPopAttrib(); } } @Override public void drawBG( int offsetX, int offsetY, int mouseX, int mouseY ) { this.bindTexture( "guis/craftingcpu.png" ); this.drawTexturedModalRect( offsetX, offsetY, 0, 0, this.xSize, this.ySize ); } public void postUpdate( List list, byte ref ) { switch( ref ) { case 0: for( IAEItemStack l : list ) { this.handleInput( this.storage, l ); } break; case 1: for( IAEItemStack l : list ) { this.handleInput( this.active, l ); } break; case 2: for( IAEItemStack l : list ) { this.handleInput( this.pending, l ); } break; } for( IAEItemStack l : list ) { long amt = this.getTotal( l ); if( amt <= 0 ) { this.deleteVisualStack( l ); } else { IAEItemStack is = this.findVisualStack( l ); is.setStackSize( amt ); } } this.setScrollBar(); } private void handleInput( IItemList s, IAEItemStack l ) { IAEItemStack a = s.findPrecise( l ); if( l.getStackSize() <= 0 ) { if( a != null ) { a.reset(); } } else { if( a == null ) { s.add( l.copy() ); a = s.findPrecise( l ); } if( a != null ) { a.setStackSize( l.getStackSize() ); } } } private long getTotal( IAEItemStack is ) { final IAEItemStack a = this.storage.findPrecise( is ); final IAEItemStack b = this.active.findPrecise( is ); final IAEItemStack c = this.pending.findPrecise( is ); long total = 0; if( a != null ) { total += a.getStackSize(); } if( b != null ) { total += b.getStackSize(); } if( c != null ) { total += c.getStackSize(); } return total; } private void deleteVisualStack( IAEItemStack l ) { final Iterator i = this.visual.iterator(); while( i.hasNext() ) { IAEItemStack o = i.next(); if( o.equals( l ) ) { i.remove(); return; } } } private IAEItemStack findVisualStack( IAEItemStack l ) { for( IAEItemStack o : this.visual ) { if( o.equals( l ) ) { return o; } } final IAEItemStack stack = l.copy(); this.visual.add( stack ); return stack; } @Override public Enum getSortBy() { return SortOrder.NAME; } @Override public Enum getSortDir() { return SortDir.ASCENDING; } @Override public Enum getSortDisplay() { return ViewItems.ALL; } }