mirror of
				https://github.com/SquidDev-CC/CC-Tweaked
				synced 2025-10-26 03:17:38 +00:00 
			
		
		
		
	Update CC: Tweaked to 1.13
Look, I originally had this split into several commits, but lots of
other cleanups got mixed in. I then backported some of the cleanups to
1.12, did other tidy ups there, and eventually the web of merges was
unreadable.
Yes, this is a horrible mess, but it's still nicer than it was. Anyway,
changes:
 - Flatten everything. For instance, there are now three instances of
   BlockComputer, two BlockTurtle, ItemPocketComputer. There's also no
   more BlockPeripheral (thank heavens) - there's separate block classes
   for each peripheral type.
 - Remove pretty much all legacy code. As we're breaking world
   compatibility anyway, we can remove all the code to load worlds from
   1.4 days.
 - The command system is largely rewriten to take advantage of 1.13's
   new system. It's very fancy!
 - WidgetTerminal now uses Minecraft's "GUI listener" system.
 - BREAKING CHANGE: All the codes in keys.lua are different, due to the
   move to LWJGL 3. Hopefully this won't have too much of an impact.
   I don't want to map to the old key codes on the Java side, as there
   always ends up being small but slight inconsistencies. IMO it's
   better to make a clean break - people should be using keys rather
   than hard coding the constants anyway.
 - commands.list now allows fetching sub-commands. The ROM has already
   been updated to allow fancy usage such as commands.time.set("noon").
 - Turtles, modems and cables can be waterlogged.
			
			
This commit is contained in:
		
							
								
								
									
										99
									
								
								build.gradle
									
									
									
									
									
								
							
							
						
						
									
										99
									
								
								build.gradle
									
									
									
									
									
								
							| @@ -1,8 +1,7 @@ | |||||||
|  |  | ||||||
| // For those who want the bleeding edge |  | ||||||
| buildscript { | buildscript { | ||||||
|     repositories { |     repositories { | ||||||
|         jcenter() |         jcenter() | ||||||
|  |         mavenCentral() | ||||||
|         maven { |         maven { | ||||||
|             name = "forge" |             name = "forge" | ||||||
|             url = "http://files.minecraftforge.net/maven" |             url = "http://files.minecraftforge.net/maven" | ||||||
| @@ -10,17 +9,17 @@ buildscript { | |||||||
|     } |     } | ||||||
|     dependencies { |     dependencies { | ||||||
|         classpath 'com.google.code.gson:gson:2.8.1' |         classpath 'com.google.code.gson:gson:2.8.1' | ||||||
|         classpath 'net.minecraftforge.gradle:ForgeGradle:2.3-SNAPSHOT' |         classpath 'net.minecraftforge.gradle:ForgeGradle:3.0.115' | ||||||
|         classpath 'net.sf.proguard:proguard-gradle:6.1.0beta1' |         classpath 'net.sf.proguard:proguard-gradle:6.1.0beta2' | ||||||
|         classpath 'org.ajoberstar.grgit:grgit-gradle:3.0.0' |         classpath 'org.ajoberstar.grgit:grgit-gradle:3.0.0' | ||||||
|     } |     } | ||||||
| } | } | ||||||
|  |  | ||||||
| plugins { | plugins { | ||||||
|     id 'com.matthewprenger.cursegradle' version '1.0.10' |     id 'com.matthewprenger.cursegradle' version '1.2.0' | ||||||
| } | } | ||||||
|  |  | ||||||
| apply plugin: 'net.minecraftforge.gradle.forge' | apply plugin: 'net.minecraftforge.gradle' | ||||||
| apply plugin: 'org.ajoberstar.grgit' | apply plugin: 'org.ajoberstar.grgit' | ||||||
| apply plugin: 'maven-publish' | apply plugin: 'maven-publish' | ||||||
| apply plugin: 'maven' | apply plugin: 'maven' | ||||||
| @@ -31,18 +30,41 @@ group = "org.squiddev" | |||||||
| archivesBaseName = "cc-tweaked-${mc_version}" | archivesBaseName = "cc-tweaked-${mc_version}" | ||||||
|  |  | ||||||
| minecraft { | minecraft { | ||||||
|     version = "${mc_version}-${forge_version}" |     runs { | ||||||
|     runDir = "run" |         client { | ||||||
|     replace '${version}', mod_version |             workingDirectory project.file('run') | ||||||
|  |             property 'forge.logging.markers', 'REGISTRIES' | ||||||
|  |             property 'forge.logging.console.level', 'debug' | ||||||
|  |  | ||||||
|     mappings = mappings_version |             mods { | ||||||
|     makeObfSourceJar = false |                 computercraft { | ||||||
|  |                     source sourceSets.main | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         server { | ||||||
|  |             workingDirectory project.file('run') | ||||||
|  |             property 'forge.logging.markers', 'SCAN,REGISTRIES,REGISTRYDUMP' | ||||||
|  |             property 'forge.logging.console.level', 'debug' | ||||||
|  |  | ||||||
|  |             mods { | ||||||
|  |                 computercraft { | ||||||
|  |                     source sourceSets.main | ||||||
|  |                 } | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     mappings channel: 'snapshot', version: "${mappings_version}".toString() | ||||||
|  |  | ||||||
|  |     accessTransformer file('src/main/resources/META-INF/accesstransformer.cfg') | ||||||
| } | } | ||||||
|  |  | ||||||
| repositories { | repositories { | ||||||
|     maven { |     maven { | ||||||
|         name "JEI" |         name "JEI" | ||||||
|         url "http://dvs1.progwml6.com/files/maven" |         url  "http://dvs1.progwml6.com/files/maven" | ||||||
|     } |     } | ||||||
|     maven { |     maven { | ||||||
|         name "SquidDev" |         name "SquidDev" | ||||||
| @@ -65,11 +87,13 @@ configurations { | |||||||
| } | } | ||||||
|  |  | ||||||
| dependencies { | dependencies { | ||||||
|     deobfProvided "mezz.jei:jei_1.12.2:4.15.0.269:api" |     minecraft "net.minecraftforge:forge:${mc_version}-${forge_version}" | ||||||
|     deobfProvided "pl.asie:Charset-Lib:0.5.4.6" |  | ||||||
|     deobfProvided "MCMultiPart2:MCMultiPart:2.5.3" |  | ||||||
|  |  | ||||||
|     runtime "mezz.jei:jei_1.12.2:4.15.0.269" |     compileOnly "mezz.jei:jei-1.13.2:5.0.0.8:api" | ||||||
|  |     // deobfProvided "pl.asie:Charset-Lib:0.5.4.6" | ||||||
|  |     // deobfProvided "MCMultiPart2:MCMultiPart:2.5.3" | ||||||
|  |  | ||||||
|  |     deobf "mezz.jei:jei-1.13.2:5.0.0.8" | ||||||
|  |  | ||||||
|     shade 'org.squiddev:Cobalt:0.5.0-SNAPSHOT' |     shade 'org.squiddev:Cobalt:0.5.0-SNAPSHOT' | ||||||
|  |  | ||||||
| @@ -79,6 +103,15 @@ dependencies { | |||||||
|     deployerJars "org.apache.maven.wagon:wagon-ssh:3.0.0" |     deployerJars "org.apache.maven.wagon:wagon-ssh:3.0.0" | ||||||
| } | } | ||||||
|  |  | ||||||
|  | sourceSets { | ||||||
|  |     main { | ||||||
|  |         java { | ||||||
|  |             exclude 'dan200/computercraft/shared/integration/mcmp' | ||||||
|  |             exclude 'dan200/computercraft/shared/integration/charset' | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | } | ||||||
|  |  | ||||||
| javadoc { | javadoc { | ||||||
|     include "dan200/computercraft/api/**/*.java" |     include "dan200/computercraft/api/**/*.java" | ||||||
| } | } | ||||||
| @@ -87,7 +120,13 @@ jar { | |||||||
|     dependsOn javadoc |     dependsOn javadoc | ||||||
|  |  | ||||||
|     manifest { |     manifest { | ||||||
|         attributes('FMLAT': 'computercraft_at.cfg') |         attributes(["Specification-Title": "computercraft", | ||||||
|  |                     "Specification-Vendor": "SquidDev", | ||||||
|  |                     "Specification-Version": "25.0", | ||||||
|  |                     "Implementation-Title": "CC: Tweaked", | ||||||
|  |                     "Implementation-Version": "${mod_version}", | ||||||
|  |                     "Implementation-Vendor" :"SquidDev", | ||||||
|  |                     "Implementation-Timestamp": new Date().format("yyyy-MM-dd'T'HH:mm:ssZ")]) | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     from (sourceSets.main.allSource) { |     from (sourceSets.main.allSource) { | ||||||
| @@ -132,6 +171,10 @@ task proguard(type: ProGuardTask, dependsOn: jar) { | |||||||
|  |  | ||||||
|     // Preserve the constructors in Cobalt library class, as we init them via reflection |     // Preserve the constructors in Cobalt library class, as we init them via reflection | ||||||
|     keepclassmembers 'class org.squiddev.cobalt.lib.** { <init>(...); }' |     keepclassmembers 'class org.squiddev.cobalt.lib.** { <init>(...); }' | ||||||
|  |  | ||||||
|  |     // LWJGL and Apache bundle Java 9 versions, which is great, but rather breaks Proguard | ||||||
|  |     dontwarn 'module-info' | ||||||
|  |     dontwarn 'org.apache.**,org.lwjgl.**' | ||||||
| } | } | ||||||
|  |  | ||||||
| task proguardMove(dependsOn: proguard) { | task proguardMove(dependsOn: proguard) { | ||||||
| @@ -147,7 +190,7 @@ task proguardMove(dependsOn: proguard) { | |||||||
|     } |     } | ||||||
| } | } | ||||||
|  |  | ||||||
| reobfJar.dependsOn proguardMove |  | ||||||
|  |  | ||||||
| processResources { | processResources { | ||||||
|     inputs.property "version", mod_version |     inputs.property "version", mod_version | ||||||
| @@ -169,8 +212,8 @@ processResources { | |||||||
|     inputs.property "commithash", hash |     inputs.property "commithash", hash | ||||||
|  |  | ||||||
|     from(sourceSets.main.resources.srcDirs) { |     from(sourceSets.main.resources.srcDirs) { | ||||||
|         include 'mcmod.info' |         include 'META-INF/mods.toml' | ||||||
|         include 'assets/computercraft/lua/rom/help/credits.txt' |         include 'data/computercraft/lua/rom/help/credits.txt' | ||||||
|  |  | ||||||
|         expand 'version': mod_version, |         expand 'version': mod_version, | ||||||
|                'mcversion': mc_version, |                'mcversion': mc_version, | ||||||
| @@ -178,12 +221,12 @@ processResources { | |||||||
|     } |     } | ||||||
|  |  | ||||||
|     from(sourceSets.main.resources.srcDirs) { |     from(sourceSets.main.resources.srcDirs) { | ||||||
|         exclude 'mcmod.info' |         exclude 'META-INF/mods.toml' | ||||||
|         exclude 'assets/computercraft/lua/rom/help/credits.txt' |         exclude 'data/computercraft/lua/rom/help/credits.txt' | ||||||
|     } |     } | ||||||
| } | } | ||||||
|  |  | ||||||
| task compressJson(dependsOn: extractAnnotationsJar) { | task compressJson(dependsOn: jar) { | ||||||
|     group "compact" |     group "compact" | ||||||
|     description "Minifies all JSON files, stripping whitespace" |     description "Minifies all JSON files, stripping whitespace" | ||||||
|  |  | ||||||
| @@ -225,6 +268,7 @@ task compressJson(dependsOn: extractAnnotationsJar) { | |||||||
|  |  | ||||||
| assemble.dependsOn compressJson | assemble.dependsOn compressJson | ||||||
|  |  | ||||||
|  | /* | ||||||
| curseforge { | curseforge { | ||||||
|     apiKey = project.hasProperty('curseForgeApiKey') ? project.curseForgeApiKey : '' |     apiKey = project.hasProperty('curseForgeApiKey') ? project.curseForgeApiKey : '' | ||||||
|     project { |     project { | ||||||
| @@ -233,12 +277,13 @@ curseforge { | |||||||
|         changelog = "Release notes can be found on the GitHub repository (https://github.com/SquidDev-CC/CC-Tweaked/releases/tag/v${project.version})." |         changelog = "Release notes can be found on the GitHub repository (https://github.com/SquidDev-CC/CC-Tweaked/releases/tag/v${project.version})." | ||||||
|     } |     } | ||||||
| } | } | ||||||
|  | */ | ||||||
|  |  | ||||||
| publishing { | publishing { | ||||||
|     publications { |     publications { | ||||||
|         mavenJava(MavenPublication) { |         mavenJava(MavenPublication) { | ||||||
|             from components.java |             from components.java | ||||||
|             artifact sourceJar |             // artifact sourceJar | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
| } | } | ||||||
| @@ -295,10 +340,10 @@ test { | |||||||
| } | } | ||||||
|  |  | ||||||
| gradle.projectsEvaluated { | gradle.projectsEvaluated { | ||||||
|  |     reobfJar.dependsOn proguardMove | ||||||
|  |  | ||||||
|     tasks.withType(JavaCompile) { |     tasks.withType(JavaCompile) { | ||||||
|         options.compilerArgs << "-Xlint" << "-Xlint:-processing" << "-Werror" |         options.compilerArgs << "-Xlint" << "-Xlint:-processing" // Causes Forge build to fail << "-Werror" | ||||||
|     } |     } | ||||||
| } | } | ||||||
|  |  | ||||||
| runClient.outputs.upToDateWhen { false } |  | ||||||
| runServer.outputs.upToDateWhen { false } |  | ||||||
|   | |||||||
| @@ -2,6 +2,6 @@ | |||||||
| mod_version=1.82.0 | mod_version=1.82.0 | ||||||
|  |  | ||||||
| # Minecraft properties | # Minecraft properties | ||||||
| mc_version=1.12.2 | mc_version=1.13.2 | ||||||
| forge_version=14.23.4.2749 | forge_version=25.0.100 | ||||||
| mappings_version=snapshot_20180724 | mappings_version=20190327-1.13.2 | ||||||
|   | |||||||
							
								
								
									
										
											BIN
										
									
								
								gradle/wrapper/gradle-wrapper.jar
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										
											BIN
										
									
								
								gradle/wrapper/gradle-wrapper.jar
									
									
									
									
										vendored
									
									
								
							
										
											Binary file not shown.
										
									
								
							
							
								
								
									
										2
									
								
								gradle/wrapper/gradle-wrapper.properties
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										2
									
								
								gradle/wrapper/gradle-wrapper.properties
									
									
									
									
										vendored
									
									
								
							| @@ -1,5 +1,5 @@ | |||||||
| distributionBase=GRADLE_USER_HOME | distributionBase=GRADLE_USER_HOME | ||||||
| distributionPath=wrapper/dists | distributionPath=wrapper/dists | ||||||
|  | distributionUrl=https\://services.gradle.org/distributions/gradle-4.9-bin.zip | ||||||
| zipStoreBase=GRADLE_USER_HOME | zipStoreBase=GRADLE_USER_HOME | ||||||
| zipStorePath=wrapper/dists | zipStorePath=wrapper/dists | ||||||
| distributionUrl=https\://services.gradle.org/distributions/gradle-4.6-bin.zip |  | ||||||
|   | |||||||
| @@ -1,30 +0,0 @@ | |||||||
| /* |  | ||||||
|  * This file is part of ComputerCraft - http://www.computercraft.info |  | ||||||
|  * Copyright Daniel Ratcliffe, 2011-2019. Do not distribute without permission. |  | ||||||
|  * Send enquiries to dratcliffe@gmail.com |  | ||||||
|  */ |  | ||||||
|  |  | ||||||
| package dan200.computercraft; |  | ||||||
|  |  | ||||||
| import net.minecraftforge.fml.common.Mod; |  | ||||||
| import net.minecraftforge.fml.common.network.NetworkCheckHandler; |  | ||||||
| import net.minecraftforge.fml.relauncher.Side; |  | ||||||
|  |  | ||||||
| import java.util.Map; |  | ||||||
|  |  | ||||||
| /** |  | ||||||
|  * A stub mod for CC: Tweaked. This doesn't have any functionality (everything of note is done in |  | ||||||
|  * {@link ComputerCraft}), but people may depend on this if they require CC: Tweaked functionality. |  | ||||||
|  */ |  | ||||||
| @Mod( |  | ||||||
|     modid = "cctweaked", name = ComputerCraft.NAME, version = ComputerCraft.VERSION, |  | ||||||
|     acceptableRemoteVersions = "*" |  | ||||||
| ) |  | ||||||
| public class CCTweaked |  | ||||||
| { |  | ||||||
|     @NetworkCheckHandler |  | ||||||
|     public boolean onNetworkConnect( Map<String, String> mods, Side side ) |  | ||||||
|     { |  | ||||||
|         return true; |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| @@ -7,99 +7,50 @@ | |||||||
| package dan200.computercraft; | package dan200.computercraft; | ||||||
|  |  | ||||||
| import dan200.computercraft.api.filesystem.IMount; | import dan200.computercraft.api.filesystem.IMount; | ||||||
| import dan200.computercraft.api.filesystem.IWritableMount; |  | ||||||
| import dan200.computercraft.api.lua.ILuaAPIFactory; |  | ||||||
| import dan200.computercraft.api.media.IMedia; |  | ||||||
| import dan200.computercraft.api.media.IMediaProvider; |  | ||||||
| import dan200.computercraft.api.network.IPacketNetwork; |  | ||||||
| import dan200.computercraft.api.network.wired.IWiredElement; |  | ||||||
| import dan200.computercraft.api.network.wired.IWiredNode; |  | ||||||
| import dan200.computercraft.api.peripheral.IPeripheral; |  | ||||||
| import dan200.computercraft.api.peripheral.IPeripheralProvider; |  | ||||||
| import dan200.computercraft.api.permissions.ITurtlePermissionProvider; |  | ||||||
| import dan200.computercraft.api.pocket.IPocketUpgrade; |  | ||||||
| import dan200.computercraft.api.redstone.IBundledRedstoneProvider; |  | ||||||
| import dan200.computercraft.api.turtle.ITurtleUpgrade; |  | ||||||
| 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.ApiFactories; |  | ||||||
| import dan200.computercraft.core.apis.http.websocket.Websocket; | import dan200.computercraft.core.apis.http.websocket.Websocket; | ||||||
| import dan200.computercraft.core.computer.MainThread; | import dan200.computercraft.core.filesystem.ResourceMount; | ||||||
| import dan200.computercraft.core.filesystem.ComboMount; | import dan200.computercraft.shared.Config; | ||||||
| import dan200.computercraft.core.filesystem.FileMount; |  | ||||||
| import dan200.computercraft.core.filesystem.JarMount; |  | ||||||
| import dan200.computercraft.core.tracking.Tracking; |  | ||||||
| import dan200.computercraft.shared.*; |  | ||||||
| import dan200.computercraft.shared.computer.blocks.BlockCommandComputer; |  | ||||||
| import dan200.computercraft.shared.computer.blocks.BlockComputer; | import dan200.computercraft.shared.computer.blocks.BlockComputer; | ||||||
| import dan200.computercraft.shared.computer.core.ClientComputerRegistry; | import dan200.computercraft.shared.computer.core.ClientComputerRegistry; | ||||||
| import dan200.computercraft.shared.computer.core.ServerComputerRegistry; | import dan200.computercraft.shared.computer.core.ServerComputerRegistry; | ||||||
| import dan200.computercraft.shared.computer.items.ItemCommandComputer; |  | ||||||
| import dan200.computercraft.shared.computer.items.ItemComputer; | import dan200.computercraft.shared.computer.items.ItemComputer; | ||||||
| import dan200.computercraft.shared.media.items.ItemDiskExpanded; | import dan200.computercraft.shared.media.items.ItemDisk; | ||||||
| import dan200.computercraft.shared.media.items.ItemDiskLegacy; |  | ||||||
| import dan200.computercraft.shared.media.items.ItemPrintout; | import dan200.computercraft.shared.media.items.ItemPrintout; | ||||||
| import dan200.computercraft.shared.media.items.ItemTreasureDisk; | import dan200.computercraft.shared.media.items.ItemTreasureDisk; | ||||||
| import dan200.computercraft.shared.peripheral.common.BlockPeripheral; | import dan200.computercraft.shared.peripheral.diskdrive.BlockDiskDrive; | ||||||
| import dan200.computercraft.shared.peripheral.common.ItemPeripheral; |  | ||||||
| import dan200.computercraft.shared.peripheral.modem.wired.BlockCable; | import dan200.computercraft.shared.peripheral.modem.wired.BlockCable; | ||||||
| import dan200.computercraft.shared.peripheral.modem.wired.BlockWiredModemFull; | import dan200.computercraft.shared.peripheral.modem.wired.BlockWiredModemFull; | ||||||
| import dan200.computercraft.shared.peripheral.modem.wired.ItemCable; | import dan200.computercraft.shared.peripheral.modem.wired.ItemBlockCable; | ||||||
| import dan200.computercraft.shared.peripheral.modem.wireless.BlockAdvancedModem; | import dan200.computercraft.shared.peripheral.modem.wireless.BlockWirelessModem; | ||||||
| import dan200.computercraft.shared.peripheral.modem.wireless.ItemAdvancedModem; | import dan200.computercraft.shared.peripheral.monitor.BlockMonitor; | ||||||
| import dan200.computercraft.shared.peripheral.modem.wireless.WirelessNetwork; | import dan200.computercraft.shared.peripheral.printer.BlockPrinter; | ||||||
|  | import dan200.computercraft.shared.peripheral.speaker.BlockSpeaker; | ||||||
| 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; | ||||||
| import dan200.computercraft.shared.proxy.ComputerCraftProxyCommon; |  | ||||||
| import dan200.computercraft.shared.turtle.blocks.BlockTurtle; | import dan200.computercraft.shared.turtle.blocks.BlockTurtle; | ||||||
| import dan200.computercraft.shared.turtle.items.ItemTurtleAdvanced; | import dan200.computercraft.shared.turtle.items.ItemTurtle; | ||||||
| import dan200.computercraft.shared.turtle.items.ItemTurtleLegacy; |  | ||||||
| import dan200.computercraft.shared.turtle.items.ItemTurtleNormal; |  | ||||||
| import dan200.computercraft.shared.turtle.upgrades.*; | import dan200.computercraft.shared.turtle.upgrades.*; | ||||||
| import dan200.computercraft.shared.util.CreativeTabMain; | import net.minecraft.resources.IReloadableResourceManager; | ||||||
| import dan200.computercraft.shared.util.IDAssigner; | import net.minecraft.util.ResourceLocation; | ||||||
| import dan200.computercraft.shared.util.IoUtil; |  | ||||||
| import dan200.computercraft.shared.wired.CapabilityWiredElement; |  | ||||||
| import dan200.computercraft.shared.wired.WiredNode; |  | ||||||
| import net.minecraft.entity.player.EntityPlayer; |  | ||||||
| import net.minecraft.item.ItemBlock; |  | ||||||
| import net.minecraft.item.ItemStack; |  | ||||||
| import net.minecraft.server.MinecraftServer; |  | ||||||
| import net.minecraft.tileentity.TileEntity; |  | ||||||
| import net.minecraft.util.EnumFacing; |  | ||||||
| import net.minecraft.util.math.BlockPos; |  | ||||||
| import net.minecraft.world.IBlockAccess; |  | ||||||
| import net.minecraft.world.World; |  | ||||||
| import net.minecraftforge.common.DimensionManager; |  | ||||||
| import net.minecraftforge.fml.common.FMLCommonHandler; |  | ||||||
| import net.minecraftforge.fml.common.Mod; | import net.minecraftforge.fml.common.Mod; | ||||||
| import net.minecraftforge.fml.common.SidedProxy; | import net.minecraftforge.fml.server.ServerLifecycleHooks; | ||||||
| import net.minecraftforge.fml.common.event.*; | import org.apache.logging.log4j.LogManager; | ||||||
| import net.minecraftforge.fml.relauncher.Side; |  | ||||||
| import org.apache.logging.log4j.Logger; | import org.apache.logging.log4j.Logger; | ||||||
|  |  | ||||||
| import java.io.*; | import java.io.IOException; | ||||||
| import java.net.MalformedURLException; | import java.io.InputStream; | ||||||
| import java.net.URISyntaxException; |  | ||||||
| import java.net.URL; |  | ||||||
| import java.util.ArrayList; |  | ||||||
| import java.util.EnumSet; | import java.util.EnumSet; | ||||||
| import java.util.List; |  | ||||||
| import java.util.concurrent.TimeUnit; | import java.util.concurrent.TimeUnit; | ||||||
| import java.util.zip.ZipEntry; |  | ||||||
| import java.util.zip.ZipFile; |  | ||||||
|  |  | ||||||
| @Mod( | @Mod( ComputerCraft.MOD_ID ) | ||||||
|     modid = ComputerCraft.MOD_ID, name = ComputerCraft.NAME, version = ComputerCraft.VERSION, | public final class ComputerCraft | ||||||
|     guiFactory = "dan200.computercraft.client.gui.GuiConfigCC$Factory", |  | ||||||
|     dependencies = "required:forge@[14.23.4.2746,)" |  | ||||||
| ) |  | ||||||
| public class ComputerCraft |  | ||||||
| { | { | ||||||
|     public static final String MOD_ID = "computercraft"; |     public static final String MOD_ID = "computercraft"; | ||||||
|     static final String VERSION = "${version}"; |  | ||||||
|     static final String NAME = "CC: Tweaked"; |     public static final int DATAFIXER_VERSION = 0; | ||||||
|  |  | ||||||
|     // Configuration options |     // Configuration options | ||||||
|     public static final String[] DEFAULT_HTTP_WHITELIST = new String[] { "*" }; |     public static final String[] DEFAULT_HTTP_WHITELIST = new String[] { "*" }; | ||||||
| @@ -161,46 +112,54 @@ public class ComputerCraft | |||||||
|     // Blocks and Items |     // Blocks and Items | ||||||
|     public static final class Blocks |     public static final class Blocks | ||||||
|     { |     { | ||||||
|         public static BlockComputer computer; |         public static BlockComputer computerNormal; | ||||||
|         public static BlockCommandComputer commandComputer; |         public static BlockComputer computerAdvanced; | ||||||
|  |         public static BlockComputer computerCommand; | ||||||
|  |  | ||||||
|         public static BlockTurtle turtle; |         public static BlockTurtle turtleNormal; | ||||||
|         public static BlockTurtle turtleExpanded; |  | ||||||
|         public static BlockTurtle turtleAdvanced; |         public static BlockTurtle turtleAdvanced; | ||||||
|  |  | ||||||
|         public static BlockPeripheral peripheral; |         public static BlockSpeaker speaker; | ||||||
|         public static BlockCable cable; |         public static BlockDiskDrive diskDrive; | ||||||
|         public static BlockAdvancedModem advancedModem; |         public static BlockPrinter printer; | ||||||
|  |  | ||||||
|  |         public static BlockMonitor monitorNormal; | ||||||
|  |         public static BlockMonitor monitorAdvanced; | ||||||
|  |  | ||||||
|  |         public static BlockWirelessModem wirelessModemNormal; | ||||||
|  |         public static BlockWirelessModem wirelessModemAdvanced; | ||||||
|  |  | ||||||
|         public static BlockWiredModemFull wiredModemFull; |         public static BlockWiredModemFull wiredModemFull; | ||||||
|  |         public static BlockCable cable; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     public static final class Items |     public static final class Items | ||||||
|     { |     { | ||||||
|         public static ItemComputer computer; |         public static ItemComputer computerNormal; | ||||||
|         public static ItemCommandComputer commandComputer; |         public static ItemComputer computerAdvanced; | ||||||
|  |         public static ItemComputer computerCommand; | ||||||
|  |  | ||||||
|         public static ItemTurtleLegacy turtle; |         public static ItemPocketComputer pocketComputerNormal; | ||||||
|         public static ItemTurtleNormal turtleExpanded; |         public static ItemPocketComputer pocketComputerAdvanced; | ||||||
|         public static ItemTurtleAdvanced turtleAdvanced; |  | ||||||
|  |  | ||||||
|         public static ItemPocketComputer pocketComputer; |         public static ItemTurtle turtleNormal; | ||||||
|  |         public static ItemTurtle turtleAdvanced; | ||||||
|  |  | ||||||
|         public static ItemDiskLegacy disk; |         public static ItemDisk disk; | ||||||
|         public static ItemDiskExpanded diskExpanded; |  | ||||||
|         public static ItemTreasureDisk treasureDisk; |         public static ItemTreasureDisk treasureDisk; | ||||||
|  |  | ||||||
|         public static ItemPrintout printout; |         public static ItemPrintout printedPage; | ||||||
|  |         public static ItemPrintout printedPages; | ||||||
|  |         public static ItemPrintout printedBook; | ||||||
|  |  | ||||||
|         public static ItemPeripheral peripheral; |         public static ItemBlockCable.Cable cable; | ||||||
|         public static ItemAdvancedModem advancedModem; |         public static ItemBlockCable.WiredModem wiredModem; | ||||||
|         public static ItemCable cable; |  | ||||||
|         public static ItemBlock wiredModemFull; |  | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     public static final class TurtleUpgrades |     public static final class TurtleUpgrades | ||||||
|     { |     { | ||||||
|         public static TurtleModem wirelessModem; |         public static TurtleModem wirelessModemNormal; | ||||||
|         public static TurtleModem advancedModem; |         public static TurtleModem wirelessModemAdvanced; | ||||||
|         public static TurtleSpeaker speaker; |         public static TurtleSpeaker speaker; | ||||||
|  |  | ||||||
|         public static TurtleCraftingTable craftingTable; |         public static TurtleCraftingTable craftingTable; | ||||||
| @@ -213,445 +172,45 @@ public class ComputerCraft | |||||||
|  |  | ||||||
|     public static final class PocketUpgrades |     public static final class PocketUpgrades | ||||||
|     { |     { | ||||||
|         public static PocketModem wirelessModem; |         public static PocketModem wirelessModemNormal; | ||||||
|         public static PocketModem advancedModem; |         public static PocketModem wirelessModemAdvanced; | ||||||
|         public static PocketSpeaker speaker; |         public static PocketSpeaker speaker; | ||||||
|  |  | ||||||
|         @Deprecated |  | ||||||
|         public static PocketSpeaker pocketSpeaker; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     @Deprecated |  | ||||||
|     public static final class Upgrades |  | ||||||
|     { |  | ||||||
|         public static TurtleModem advancedModem; |  | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     // Registries |     // Registries | ||||||
|     public static final ClientComputerRegistry clientComputerRegistry = new ClientComputerRegistry(); |     public static final ClientComputerRegistry clientComputerRegistry = new ClientComputerRegistry(); | ||||||
|     public static final ServerComputerRegistry serverComputerRegistry = new ServerComputerRegistry(); |     public static final ServerComputerRegistry serverComputerRegistry = new ServerComputerRegistry(); | ||||||
|  |  | ||||||
|     // Creative |  | ||||||
|     public static CreativeTabMain mainCreativeTab; |  | ||||||
|  |  | ||||||
|     // Logging |     // Logging | ||||||
|     public static Logger log; |     public static final Logger log = LogManager.getLogger( MOD_ID ); | ||||||
|  |  | ||||||
|     // Peripheral providers. This is still here to ensure compatibility with Plethora and Computronics |     public ComputerCraft() | ||||||
|     public static List<IPeripheralProvider> peripheralProviders = new ArrayList<>(); |  | ||||||
|  |  | ||||||
|     // Implementation |  | ||||||
|     @Mod.Instance( ComputerCraft.MOD_ID ) |  | ||||||
|     public static ComputerCraft instance; |  | ||||||
|  |  | ||||||
|     @SidedProxy( |  | ||||||
|         clientSide = "dan200.computercraft.client.proxy.ComputerCraftProxyClient", |  | ||||||
|         serverSide = "dan200.computercraft.shared.proxy.ComputerCraftProxyCommon" |  | ||||||
|     ) |  | ||||||
|     private static ComputerCraftProxyCommon proxy; |  | ||||||
|  |  | ||||||
|     @Mod.EventHandler |  | ||||||
|     public void preInit( FMLPreInitializationEvent event ) |  | ||||||
|     { |     { | ||||||
|         log = event.getModLog(); |         Config.load(); | ||||||
|  |  | ||||||
|         // Load config |  | ||||||
|         Config.load( event.getSuggestedConfigurationFile() ); |  | ||||||
|  |  | ||||||
|         proxy.preInit(); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     @Mod.EventHandler |  | ||||||
|     public void init( FMLInitializationEvent event ) |  | ||||||
|     { |  | ||||||
|         proxy.init(); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     @Mod.EventHandler |  | ||||||
|     public void onServerStarting( FMLServerStartingEvent event ) |  | ||||||
|     { |  | ||||||
|         ComputerCraftProxyCommon.initServer( event.getServer() ); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     @Mod.EventHandler |  | ||||||
|     public void onServerStart( FMLServerStartedEvent event ) |  | ||||||
|     { |  | ||||||
|         if( FMLCommonHandler.instance().getEffectiveSide() == Side.SERVER ) |  | ||||||
|         { |  | ||||||
|             ComputerCraft.serverComputerRegistry.reset(); |  | ||||||
|             WirelessNetwork.resetNetworks(); |  | ||||||
|             MainThread.reset(); |  | ||||||
|             Tracking.reset(); |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     @Mod.EventHandler |  | ||||||
|     public void onServerStopped( FMLServerStoppedEvent event ) |  | ||||||
|     { |  | ||||||
|         if( FMLCommonHandler.instance().getEffectiveSide() == Side.SERVER ) |  | ||||||
|         { |  | ||||||
|             ComputerCraft.serverComputerRegistry.reset(); |  | ||||||
|             WirelessNetwork.resetNetworks(); |  | ||||||
|             MainThread.reset(); |  | ||||||
|             Tracking.reset(); |  | ||||||
|         } |  | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     public static String getVersion() |     public static String getVersion() | ||||||
|     { |     { | ||||||
|         return VERSION; |         return "${version}"; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     private static File getBaseDir() |     static IMount createResourceMount( String domain, String subPath ) | ||||||
|     { |     { | ||||||
|         return FMLCommonHandler.instance().getMinecraftServerInstance().getDataDirectory(); |         IReloadableResourceManager manager = ServerLifecycleHooks.getCurrentServer().getResourceManager(); | ||||||
|  |         ResourceMount mount = new ResourceMount( domain, subPath, manager ); | ||||||
|  |         return mount.exists( "" ) ? mount : null; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     private static File getResourcePackDir() |     public static InputStream getResourceFile( String domain, String subPath ) | ||||||
|     { |  | ||||||
|         return new File( getBaseDir(), "resourcepacks" ); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     @Deprecated |  | ||||||
|     public static void registerPermissionProvider( ITurtlePermissionProvider provider ) |  | ||||||
|     { |  | ||||||
|         TurtlePermissions.register( provider ); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     @Deprecated |  | ||||||
|     public static void registerPocketUpgrade( IPocketUpgrade upgrade ) |  | ||||||
|     { |  | ||||||
|         dan200.computercraft.shared.PocketUpgrades.register( upgrade ); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     @Deprecated |  | ||||||
|     public static void registerPeripheralProvider( IPeripheralProvider provider ) |  | ||||||
|     { |  | ||||||
|         Peripherals.register( provider ); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     @Deprecated |  | ||||||
|     public static void registerBundledRedstoneProvider( IBundledRedstoneProvider provider ) |  | ||||||
|     { |  | ||||||
|         BundledRedstone.register( provider ); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     @Deprecated |  | ||||||
|     public static void registerMediaProvider( IMediaProvider provider ) |  | ||||||
|     { |  | ||||||
|         MediaProviders.register( provider ); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     @Deprecated |  | ||||||
|     public static void registerAPIFactory( ILuaAPIFactory factory ) |  | ||||||
|     { |  | ||||||
|         ApiFactories.register( factory ); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     @Deprecated |  | ||||||
|     public static IWiredNode createWiredNodeForElement( IWiredElement element ) |  | ||||||
|     { |  | ||||||
|         return new WiredNode( element ); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     @Deprecated |  | ||||||
|     public static IWiredElement getWiredElementAt( IBlockAccess world, BlockPos pos, EnumFacing side ) |  | ||||||
|     { |  | ||||||
|         TileEntity tile = world.getTileEntity( pos ); |  | ||||||
|         return tile != null && tile.hasCapability( CapabilityWiredElement.CAPABILITY, side ) |  | ||||||
|             ? tile.getCapability( CapabilityWiredElement.CAPABILITY, side ) |  | ||||||
|             : null; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     @Deprecated |  | ||||||
|     public static int getDefaultBundledRedstoneOutput( World world, BlockPos pos, EnumFacing side ) |  | ||||||
|     { |  | ||||||
|         return BundledRedstone.getDefaultOutput( world, pos, side ); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     @Deprecated |  | ||||||
|     public static IPacketNetwork getWirelessNetwork() |  | ||||||
|     { |  | ||||||
|         return WirelessNetwork.getUniversal(); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     @Deprecated |  | ||||||
|     public static int createUniqueNumberedSaveDir( World world, String parentSubPath ) |  | ||||||
|     { |  | ||||||
|         return IDAssigner.getNextIDFromDirectory( parentSubPath ); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     @Deprecated |  | ||||||
|     public static IWritableMount createSaveDirMount( World world, String subPath, long capacity ) |  | ||||||
|     { |     { | ||||||
|  |         IReloadableResourceManager manager = ServerLifecycleHooks.getCurrentServer().getResourceManager(); | ||||||
|         try |         try | ||||||
|         { |         { | ||||||
|             return new FileMount( new File( getWorldDir(), subPath ), capacity ); |             return manager.getResource( new ResourceLocation( domain, subPath ) ).getInputStream(); | ||||||
|         } |         } | ||||||
|         catch( Exception e ) |         catch( IOException ignored ) | ||||||
|         { |         { | ||||||
|             return null; |             return null; | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @Deprecated |  | ||||||
|     public static IMount createResourceMount( Class<?> modClass, String domain, String subPath ) |  | ||||||
|     { |  | ||||||
|         // Start building list of mounts |  | ||||||
|         List<IMount> mounts = new ArrayList<>(); |  | ||||||
|         subPath = "assets/" + domain + "/" + subPath; |  | ||||||
|  |  | ||||||
|         // Mount from debug dir |  | ||||||
|         File codeDir = getDebugCodeDir( modClass ); |  | ||||||
|         if( codeDir != null ) |  | ||||||
|         { |  | ||||||
|             File subResource = new File( codeDir, subPath ); |  | ||||||
|             if( subResource.exists() ) |  | ||||||
|             { |  | ||||||
|                 IMount resourcePackMount = new FileMount( subResource, 0 ); |  | ||||||
|                 mounts.add( resourcePackMount ); |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         // Mount from mod jar |  | ||||||
|         File modJar = getContainingJar( modClass ); |  | ||||||
|         if( modJar != null ) |  | ||||||
|         { |  | ||||||
|             try |  | ||||||
|             { |  | ||||||
|                 mounts.add( new JarMount( modJar, subPath ) ); |  | ||||||
|             } |  | ||||||
|             catch( IOException | RuntimeException e ) |  | ||||||
|             { |  | ||||||
|                 ComputerCraft.log.error( "Could not load mount from mod jar", e ); |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         // Mount from resource packs |  | ||||||
|         File resourcePackDir = getResourcePackDir(); |  | ||||||
|         if( resourcePackDir.exists() && resourcePackDir.isDirectory() ) |  | ||||||
|         { |  | ||||||
|             String[] resourcePacks = resourcePackDir.list(); |  | ||||||
|             for( String resourcePackName : resourcePacks ) |  | ||||||
|             { |  | ||||||
|                 try |  | ||||||
|                 { |  | ||||||
|                     File resourcePack = new File( resourcePackDir, resourcePackName ); |  | ||||||
|                     if( !resourcePack.isDirectory() ) |  | ||||||
|                     { |  | ||||||
|                         // Mount a resource pack from a jar |  | ||||||
|                         mounts.add( new JarMount( resourcePack, subPath ) ); |  | ||||||
|                     } |  | ||||||
|                     else |  | ||||||
|                     { |  | ||||||
|                         // Mount a resource pack from a folder |  | ||||||
|                         File subResource = new File( resourcePack, subPath ); |  | ||||||
|                         if( subResource.exists() ) mounts.add( new FileMount( subResource, 0 ) ); |  | ||||||
|                     } |  | ||||||
|                 } |  | ||||||
|                 catch( IOException | RuntimeException e ) |  | ||||||
|                 { |  | ||||||
|                     ComputerCraft.log.error( "Could not load resource pack '" + resourcePackName + "'", e ); |  | ||||||
|                 } |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         // Return the combination of all the mounts found |  | ||||||
|         if( mounts.size() >= 2 ) |  | ||||||
|         { |  | ||||||
|             IMount[] mountArray = new IMount[mounts.size()]; |  | ||||||
|             mounts.toArray( mountArray ); |  | ||||||
|             return new ComboMount( mountArray ); |  | ||||||
|         } |  | ||||||
|         else if( mounts.size() == 1 ) |  | ||||||
|         { |  | ||||||
|             return mounts.get( 0 ); |  | ||||||
|         } |  | ||||||
|         else |  | ||||||
|         { |  | ||||||
|             return null; |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     public static InputStream getResourceFile( Class<?> modClass, String domain, String subPath ) |  | ||||||
|     { |  | ||||||
|         // Start searching in possible locations |  | ||||||
|         subPath = "assets/" + domain + "/" + subPath; |  | ||||||
|  |  | ||||||
|         // Look in resource packs |  | ||||||
|         File resourcePackDir = getResourcePackDir(); |  | ||||||
|         if( resourcePackDir.exists() && resourcePackDir.isDirectory() ) |  | ||||||
|         { |  | ||||||
|             String[] resourcePacks = resourcePackDir.list(); |  | ||||||
|             for( String resourcePackPath : resourcePacks ) |  | ||||||
|             { |  | ||||||
|                 File resourcePack = new File( resourcePackDir, resourcePackPath ); |  | ||||||
|                 if( resourcePack.isDirectory() ) |  | ||||||
|                 { |  | ||||||
|                     // Mount a resource pack from a folder |  | ||||||
|                     File subResource = new File( resourcePack, subPath ); |  | ||||||
|                     if( subResource.exists() && subResource.isFile() ) |  | ||||||
|                     { |  | ||||||
|                         try |  | ||||||
|                         { |  | ||||||
|                             return new FileInputStream( subResource ); |  | ||||||
|                         } |  | ||||||
|                         catch( FileNotFoundException ignored ) |  | ||||||
|                         { |  | ||||||
|                         } |  | ||||||
|                     } |  | ||||||
|                 } |  | ||||||
|                 else |  | ||||||
|                 { |  | ||||||
|                     ZipFile zipFile = null; |  | ||||||
|                     try |  | ||||||
|                     { |  | ||||||
|                         final ZipFile zip = zipFile = new ZipFile( resourcePack ); |  | ||||||
|                         ZipEntry entry = zipFile.getEntry( subPath ); |  | ||||||
|                         if( entry != null ) |  | ||||||
|                         { |  | ||||||
|                             // Return a custom InputStream which will close the original zip when finished. |  | ||||||
|                             return new FilterInputStream( zipFile.getInputStream( entry ) ) |  | ||||||
|                             { |  | ||||||
|                                 @Override |  | ||||||
|                                 public void close() throws IOException |  | ||||||
|                                 { |  | ||||||
|                                     super.close(); |  | ||||||
|                                     zip.close(); |  | ||||||
|                                 } |  | ||||||
|                             }; |  | ||||||
|                         } |  | ||||||
|                         else |  | ||||||
|                         { |  | ||||||
|                             IoUtil.closeQuietly( zipFile ); |  | ||||||
|                         } |  | ||||||
|                     } |  | ||||||
|                     catch( IOException e ) |  | ||||||
|                     { |  | ||||||
|                         if( zipFile != null ) IoUtil.closeQuietly( zipFile ); |  | ||||||
|                     } |  | ||||||
|                 } |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         // Look in debug dir |  | ||||||
|         File codeDir = getDebugCodeDir( modClass ); |  | ||||||
|         if( codeDir != null ) |  | ||||||
|         { |  | ||||||
|             File subResource = new File( codeDir, subPath ); |  | ||||||
|             if( subResource.exists() && subResource.isFile() ) |  | ||||||
|             { |  | ||||||
|                 try |  | ||||||
|                 { |  | ||||||
|                     return new FileInputStream( subResource ); |  | ||||||
|                 } |  | ||||||
|                 catch( FileNotFoundException ignored ) |  | ||||||
|                 { |  | ||||||
|                 } |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         // Look in class loader |  | ||||||
|         return modClass.getClassLoader().getResourceAsStream( subPath ); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     private static File getContainingJar( Class<?> modClass ) |  | ||||||
|     { |  | ||||||
|         String path = modClass.getProtectionDomain().getCodeSource().getLocation().getPath(); |  | ||||||
|         int bangIndex = path.indexOf( '!' ); |  | ||||||
|         if( bangIndex >= 0 ) |  | ||||||
|         { |  | ||||||
|             path = path.substring( 0, bangIndex ); |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         URL url; |  | ||||||
|         try |  | ||||||
|         { |  | ||||||
|             url = new URL( path ); |  | ||||||
|         } |  | ||||||
|         catch( MalformedURLException e1 ) |  | ||||||
|         { |  | ||||||
|             return null; |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         File file; |  | ||||||
|         try |  | ||||||
|         { |  | ||||||
|             file = new File( url.toURI() ); |  | ||||||
|         } |  | ||||||
|         catch( URISyntaxException e ) |  | ||||||
|         { |  | ||||||
|             file = new File( url.getPath() ); |  | ||||||
|         } |  | ||||||
|         return file; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     private static File getDebugCodeDir( Class<?> modClass ) |  | ||||||
|     { |  | ||||||
|         String path = modClass.getProtectionDomain().getCodeSource().getLocation().getPath(); |  | ||||||
|         int bangIndex = path.indexOf( '!' ); |  | ||||||
|         return bangIndex >= 0 ? null : new File( new File( path ).getParentFile(), "../.." ); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     @Deprecated |  | ||||||
|     public static void registerTurtleUpgrade( ITurtleUpgrade upgrade ) |  | ||||||
|     { |  | ||||||
|         dan200.computercraft.shared.TurtleUpgrades.register( upgrade ); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     public static File getWorldDir() |  | ||||||
|     { |  | ||||||
|         return DimensionManager.getCurrentSaveRootDirectory(); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     //region Compatibility |  | ||||||
|     @Deprecated |  | ||||||
|     public static File getWorldDir( World world ) |  | ||||||
|     { |  | ||||||
|         return DimensionManager.getCurrentSaveRootDirectory(); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     @Deprecated |  | ||||||
|     public static IMedia getMedia( ItemStack stack ) |  | ||||||
|     { |  | ||||||
|         return MediaProviders.get( stack ); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     @Deprecated |  | ||||||
|     public static IPocketUpgrade getPocketUpgrade( ItemStack stack ) |  | ||||||
|     { |  | ||||||
|         return dan200.computercraft.shared.PocketUpgrades.get( stack ); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     @Deprecated |  | ||||||
|     public static ITurtleUpgrade getTurtleUpgrade( ItemStack stack ) |  | ||||||
|     { |  | ||||||
|         return dan200.computercraft.shared.TurtleUpgrades.get( stack ); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     @Deprecated |  | ||||||
|     public static IPocketUpgrade getPocketUpgrade( String id ) |  | ||||||
|     { |  | ||||||
|         return dan200.computercraft.shared.PocketUpgrades.get( id ); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     @Deprecated |  | ||||||
|     public static ITurtleUpgrade getTurtleUpgrade( String id ) |  | ||||||
|     { |  | ||||||
|         return dan200.computercraft.shared.TurtleUpgrades.get( id ); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     @Deprecated |  | ||||||
|     public static IPeripheral getPeripheralAt( World world, BlockPos pos, EnumFacing side ) |  | ||||||
|     { |  | ||||||
|         return Peripherals.getPeripheral( world, pos, side ); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     @Deprecated |  | ||||||
|     public static boolean canPlayerUseCommands( EntityPlayer player ) |  | ||||||
|     { |  | ||||||
|         MinecraftServer server = player.getServer(); |  | ||||||
|         return server != null && server.getPlayerList().canSendCommands( player.getGameProfile() ); |  | ||||||
|     } |  | ||||||
|     //endregion |  | ||||||
| } | } | ||||||
|   | |||||||
							
								
								
									
										140
									
								
								src/main/java/dan200/computercraft/ComputerCraftAPIImpl.java
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										140
									
								
								src/main/java/dan200/computercraft/ComputerCraftAPIImpl.java
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,140 @@ | |||||||
|  | /* | ||||||
|  |  * This file is part of ComputerCraft - http://www.computercraft.info | ||||||
|  |  * Copyright Daniel Ratcliffe, 2011-2019. Do not distribute without permission. | ||||||
|  |  * Send enquiries to dratcliffe@gmail.com | ||||||
|  |  */ | ||||||
|  |  | ||||||
|  | package dan200.computercraft; | ||||||
|  |  | ||||||
|  | import dan200.computercraft.api.ComputerCraftAPI.IComputerCraftAPI; | ||||||
|  | import dan200.computercraft.api.filesystem.IMount; | ||||||
|  | import dan200.computercraft.api.filesystem.IWritableMount; | ||||||
|  | import dan200.computercraft.api.lua.ILuaAPIFactory; | ||||||
|  | import dan200.computercraft.api.media.IMediaProvider; | ||||||
|  | import dan200.computercraft.api.network.IPacketNetwork; | ||||||
|  | import dan200.computercraft.api.network.wired.IWiredElement; | ||||||
|  | import dan200.computercraft.api.network.wired.IWiredNode; | ||||||
|  | import dan200.computercraft.api.peripheral.IPeripheralProvider; | ||||||
|  | import dan200.computercraft.api.pocket.IPocketUpgrade; | ||||||
|  | import dan200.computercraft.api.redstone.IBundledRedstoneProvider; | ||||||
|  | import dan200.computercraft.api.turtle.ITurtleUpgrade; | ||||||
|  | import dan200.computercraft.core.apis.ApiFactories; | ||||||
|  | import dan200.computercraft.core.filesystem.FileMount; | ||||||
|  | import dan200.computercraft.shared.*; | ||||||
|  | import dan200.computercraft.shared.peripheral.modem.wireless.WirelessNetwork; | ||||||
|  | import dan200.computercraft.shared.util.IDAssigner; | ||||||
|  | import dan200.computercraft.shared.wired.CapabilityWiredElement; | ||||||
|  | import dan200.computercraft.shared.wired.WiredNode; | ||||||
|  | import net.minecraft.tileentity.TileEntity; | ||||||
|  | import net.minecraft.util.EnumFacing; | ||||||
|  | import net.minecraft.util.math.BlockPos; | ||||||
|  | import net.minecraft.world.IBlockReader; | ||||||
|  | import net.minecraft.world.World; | ||||||
|  | import net.minecraftforge.common.util.LazyOptional; | ||||||
|  |  | ||||||
|  | import javax.annotation.Nonnull; | ||||||
|  | import java.io.File; | ||||||
|  |  | ||||||
|  | public final class ComputerCraftAPIImpl implements IComputerCraftAPI | ||||||
|  | { | ||||||
|  |     public static final ComputerCraftAPIImpl INSTANCE = new ComputerCraftAPIImpl(); | ||||||
|  |  | ||||||
|  |     private ComputerCraftAPIImpl() | ||||||
|  |     { | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     @Override | ||||||
|  |     public String getInstalledVersion() | ||||||
|  |     { | ||||||
|  |         return "${version}"; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     @Override | ||||||
|  |     public int createUniqueNumberedSaveDir( @Nonnull World world, @Nonnull String parentSubPath ) | ||||||
|  |     { | ||||||
|  |         return IDAssigner.getNextId( parentSubPath ); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     @Override | ||||||
|  |     public IWritableMount createSaveDirMount( @Nonnull World world, @Nonnull String subPath, long capacity ) | ||||||
|  |     { | ||||||
|  |         try | ||||||
|  |         { | ||||||
|  |             return new FileMount( new File( IDAssigner.getDir(), subPath ), capacity ); | ||||||
|  |         } | ||||||
|  |         catch( Exception e ) | ||||||
|  |         { | ||||||
|  |             return null; | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     @Override | ||||||
|  |     public IMount createResourceMount( @Nonnull String domain, @Nonnull String subPath ) | ||||||
|  |     { | ||||||
|  |         return ComputerCraft.createResourceMount( domain, subPath ); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     @Override | ||||||
|  |     public void registerPeripheralProvider( @Nonnull IPeripheralProvider provider ) | ||||||
|  |     { | ||||||
|  |         Peripherals.register( provider ); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     @Override | ||||||
|  |     public void registerTurtleUpgrade( @Nonnull ITurtleUpgrade upgrade ) | ||||||
|  |     { | ||||||
|  |         TurtleUpgrades.register( upgrade ); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     @Override | ||||||
|  |     public void registerBundledRedstoneProvider( @Nonnull IBundledRedstoneProvider provider ) | ||||||
|  |     { | ||||||
|  |         BundledRedstone.register( provider ); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     @Override | ||||||
|  |     public int getBundledRedstoneOutput( @Nonnull World world, @Nonnull BlockPos pos, @Nonnull EnumFacing side ) | ||||||
|  |     { | ||||||
|  |         return BundledRedstone.getDefaultOutput( world, pos, side ); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     @Override | ||||||
|  |     public void registerMediaProvider( @Nonnull IMediaProvider provider ) | ||||||
|  |     { | ||||||
|  |         MediaProviders.register( provider ); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     @Override | ||||||
|  |     public void registerPocketUpgrade( @Nonnull IPocketUpgrade upgrade ) | ||||||
|  |     { | ||||||
|  |         PocketUpgrades.register( upgrade ); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     @Override | ||||||
|  |     public IPacketNetwork getWirelessNetwork() | ||||||
|  |     { | ||||||
|  |         return WirelessNetwork.getUniversal(); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     @Override | ||||||
|  |     public void registerAPIFactory( @Nonnull ILuaAPIFactory factory ) | ||||||
|  |     { | ||||||
|  |         ApiFactories.register( factory ); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     @Override | ||||||
|  |     public IWiredNode createWiredNodeForElement( @Nonnull IWiredElement element ) | ||||||
|  |     { | ||||||
|  |         return new WiredNode( element ); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     @Override | ||||||
|  |     public IWiredElement getWiredElementAt( @Nonnull IBlockReader world, @Nonnull BlockPos pos, @Nonnull EnumFacing side ) | ||||||
|  |     { | ||||||
|  |         TileEntity tile = world.getTileEntity( pos ); | ||||||
|  |         if( tile == null ) return null; | ||||||
|  |  | ||||||
|  |         LazyOptional<IWiredElement> element = tile.getCapability( CapabilityWiredElement.CAPABILITY, side ); | ||||||
|  |         return CapabilityWiredElement.unwrap( element ); | ||||||
|  |     } | ||||||
|  | } | ||||||
| @@ -8,10 +8,10 @@ package dan200.computercraft.api; | |||||||
|  |  | ||||||
| import dan200.computercraft.api.turtle.ITurtleUpgrade; | import dan200.computercraft.api.turtle.ITurtleUpgrade; | ||||||
| import dan200.computercraft.api.turtle.TurtleUpgradeType; | import dan200.computercraft.api.turtle.TurtleUpgradeType; | ||||||
| import net.minecraft.block.Block; |  | ||||||
| import net.minecraft.item.Item; |  | ||||||
| import net.minecraft.item.ItemStack; | import net.minecraft.item.ItemStack; | ||||||
|  | import net.minecraft.util.IItemProvider; | ||||||
| import net.minecraft.util.ResourceLocation; | import net.minecraft.util.ResourceLocation; | ||||||
|  | import net.minecraft.util.Util; | ||||||
|  |  | ||||||
| import javax.annotation.Nonnull; | import javax.annotation.Nonnull; | ||||||
|  |  | ||||||
| @@ -23,43 +23,31 @@ import javax.annotation.Nonnull; | |||||||
| public abstract class AbstractTurtleUpgrade implements ITurtleUpgrade | public abstract class AbstractTurtleUpgrade implements ITurtleUpgrade | ||||||
| { | { | ||||||
|     private final ResourceLocation id; |     private final ResourceLocation id; | ||||||
|     private final int legacyId; |  | ||||||
|     private final TurtleUpgradeType type; |     private final TurtleUpgradeType type; | ||||||
|     private final String adjective; |     private final String adjective; | ||||||
|     private final ItemStack stack; |     private final ItemStack stack; | ||||||
|  |  | ||||||
|     protected AbstractTurtleUpgrade( ResourceLocation id, int legacyId, TurtleUpgradeType type, String adjective, ItemStack stack ) |     protected AbstractTurtleUpgrade( ResourceLocation id, TurtleUpgradeType type, String adjective, ItemStack stack ) | ||||||
|     { |     { | ||||||
|         this.id = id; |         this.id = id; | ||||||
|         this.legacyId = legacyId; |  | ||||||
|         this.type = type; |         this.type = type; | ||||||
|         this.adjective = adjective; |         this.adjective = adjective; | ||||||
|         this.stack = stack; |         this.stack = stack; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     protected AbstractTurtleUpgrade( ResourceLocation id, int legacyId, TurtleUpgradeType type, String adjective, Item item ) |     protected AbstractTurtleUpgrade( ResourceLocation id, TurtleUpgradeType type, String adjective, IItemProvider item ) | ||||||
|     { |     { | ||||||
|         this( id, legacyId, type, adjective, new ItemStack( item ) ); |         this( id, type, adjective, new ItemStack( item ) ); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     protected AbstractTurtleUpgrade( ResourceLocation id, int legacyId, TurtleUpgradeType type, String adjective, Block block ) |     protected AbstractTurtleUpgrade( ResourceLocation id, TurtleUpgradeType type, ItemStack stack ) | ||||||
|     { |     { | ||||||
|         this( id, legacyId, type, adjective, new ItemStack( block ) ); |         this( id, type, Util.makeTranslationKey( "upgrade", id ) + ".adjective", stack ); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     protected AbstractTurtleUpgrade( ResourceLocation id, int legacyId, TurtleUpgradeType type, ItemStack stack ) |     protected AbstractTurtleUpgrade( ResourceLocation id, TurtleUpgradeType type, IItemProvider item ) | ||||||
|     { |     { | ||||||
|         this( id, legacyId, type, "upgrade." + id + ".adjective", stack ); |         this( id, type, new ItemStack( item ) ); | ||||||
|     } |  | ||||||
|  |  | ||||||
|     protected AbstractTurtleUpgrade( ResourceLocation id, int legacyId, TurtleUpgradeType type, Item item ) |  | ||||||
|     { |  | ||||||
|         this( id, legacyId, type, new ItemStack( item ) ); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     protected AbstractTurtleUpgrade( ResourceLocation id, int legacyId, TurtleUpgradeType type, Block block ) |  | ||||||
|     { |  | ||||||
|         this( id, legacyId, type, new ItemStack( block ) ); |  | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @Nonnull |     @Nonnull | ||||||
| @@ -69,12 +57,6 @@ public abstract class AbstractTurtleUpgrade implements ITurtleUpgrade | |||||||
|         return id; |         return id; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @Override |  | ||||||
|     public final int getLegacyUpgradeID() |  | ||||||
|     { |  | ||||||
|         return legacyId; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     @Nonnull |     @Nonnull | ||||||
|     @Override |     @Override | ||||||
|     public final String getUnlocalisedAdjective() |     public final String getUnlocalisedAdjective() | ||||||
|   | |||||||
| @@ -17,18 +17,16 @@ import dan200.computercraft.api.network.wired.IWiredNode; | |||||||
| import dan200.computercraft.api.peripheral.IComputerAccess; | import dan200.computercraft.api.peripheral.IComputerAccess; | ||||||
| import dan200.computercraft.api.peripheral.IPeripheral; | import dan200.computercraft.api.peripheral.IPeripheral; | ||||||
| import dan200.computercraft.api.peripheral.IPeripheralProvider; | import dan200.computercraft.api.peripheral.IPeripheralProvider; | ||||||
| import dan200.computercraft.api.permissions.ITurtlePermissionProvider; |  | ||||||
| import dan200.computercraft.api.pocket.IPocketUpgrade; | import dan200.computercraft.api.pocket.IPocketUpgrade; | ||||||
| import dan200.computercraft.api.redstone.IBundledRedstoneProvider; | import dan200.computercraft.api.redstone.IBundledRedstoneProvider; | ||||||
| import dan200.computercraft.api.turtle.ITurtleUpgrade; | import dan200.computercraft.api.turtle.ITurtleUpgrade; | ||||||
| import net.minecraft.util.EnumFacing; | import net.minecraft.util.EnumFacing; | ||||||
| import net.minecraft.util.math.BlockPos; | import net.minecraft.util.math.BlockPos; | ||||||
| import net.minecraft.world.IBlockAccess; | import net.minecraft.world.IBlockReader; | ||||||
| import net.minecraft.world.World; | import net.minecraft.world.World; | ||||||
|  |  | ||||||
| import javax.annotation.Nonnull; | import javax.annotation.Nonnull; | ||||||
| import javax.annotation.Nullable; | import javax.annotation.Nullable; | ||||||
| import java.lang.reflect.Method; |  | ||||||
|  |  | ||||||
| /** | /** | ||||||
|  * The static entry point to the ComputerCraft API. |  * The static entry point to the ComputerCraft API. | ||||||
| @@ -37,28 +35,10 @@ import java.lang.reflect.Method; | |||||||
|  */ |  */ | ||||||
| public final class ComputerCraftAPI | public final class ComputerCraftAPI | ||||||
| { | { | ||||||
|     public static boolean isInstalled() |  | ||||||
|     { |  | ||||||
|         findCC(); |  | ||||||
|         return computerCraft != null; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     @Nonnull |     @Nonnull | ||||||
|     public static String getInstalledVersion() |     public static String getInstalledVersion() | ||||||
|     { |     { | ||||||
|         findCC(); |         return getInstance().getInstalledVersion(); | ||||||
|         if( computerCraft_getVersion != null ) |  | ||||||
|         { |  | ||||||
|             try |  | ||||||
|             { |  | ||||||
|                 return (String) computerCraft_getVersion.invoke( null ); |  | ||||||
|             } |  | ||||||
|             catch( Exception e ) |  | ||||||
|             { |  | ||||||
|                 // It failed |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
|         return ""; |  | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @Nonnull |     @Nonnull | ||||||
| @@ -82,19 +62,7 @@ public final class ComputerCraftAPI | |||||||
|      */ |      */ | ||||||
|     public static int createUniqueNumberedSaveDir( @Nonnull World world, @Nonnull String parentSubPath ) |     public static int createUniqueNumberedSaveDir( @Nonnull World world, @Nonnull String parentSubPath ) | ||||||
|     { |     { | ||||||
|         findCC(); |         return getInstance().createUniqueNumberedSaveDir( world, parentSubPath ); | ||||||
|         if( computerCraft_createUniqueNumberedSaveDir != null ) |  | ||||||
|         { |  | ||||||
|             try |  | ||||||
|             { |  | ||||||
|                 return (Integer) computerCraft_createUniqueNumberedSaveDir.invoke( null, world, parentSubPath ); |  | ||||||
|             } |  | ||||||
|             catch( Exception e ) |  | ||||||
|             { |  | ||||||
|                 // It failed |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
|         return -1; |  | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
| @@ -118,55 +86,54 @@ public final class ComputerCraftAPI | |||||||
|     @Nullable |     @Nullable | ||||||
|     public static IWritableMount createSaveDirMount( @Nonnull World world, @Nonnull String subPath, long capacity ) |     public static IWritableMount createSaveDirMount( @Nonnull World world, @Nonnull String subPath, long capacity ) | ||||||
|     { |     { | ||||||
|         findCC(); |         return getInstance().createSaveDirMount( world, subPath, capacity ); | ||||||
|         if( computerCraft_createSaveDirMount != null ) |  | ||||||
|         { |  | ||||||
|             try |  | ||||||
|             { |  | ||||||
|                 return (IWritableMount) computerCraft_createSaveDirMount.invoke( null, world, subPath, capacity ); |  | ||||||
|             } |  | ||||||
|             catch( Exception e ) |  | ||||||
|             { |  | ||||||
|                 // It failed |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
|         return null; |  | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * Creates a file system mount to a resource folder, and returns it. |      * Creates a file system mount to a resource folder, and returns it. | ||||||
|      * |      * | ||||||
|      * Use in conjunction with IComputerAccess.mount() or IComputerAccess.mountWritable() to mount a resource folder |      * Use in conjunction with {@link IComputerAccess#mount} or {@link IComputerAccess#mountWritable} to mount a | ||||||
|      * onto a computer's file system. |      * resource folder onto a computer's file system. | ||||||
|      * |      * | ||||||
|      * The files in this mount will be a combination of files in the specified mod jar, and resource packs that contain |      * The files in this mount will be a combination of files in all mod jar, and data packs that contain | ||||||
|      * resources with the same domain and path. |      * resources with the same domain and path. | ||||||
|      * |      * | ||||||
|      * @param modClass A class in whose jar to look first for the resources to mount. Using your main mod class is recommended. eg: MyMod.class |      * @param domain  The domain under which to look for resources. eg: "mymod". | ||||||
|      * @param domain   The domain under which to look for resources. eg: "mymod". |      * @param subPath The subPath under which to look for resources. eg: "lua/myfiles". | ||||||
|      * @param subPath  The domain under which to look for resources. eg: "mymod/lua/myfiles". |      * @return The mount, or {@code null} if it could be created for some reason. | ||||||
|      * @return The mount, or {@code null} if it could be created for some reason. Use IComputerAccess.mount() or |  | ||||||
|      * IComputerAccess.mountWritable() to mount this on a Computers' file system. |  | ||||||
|      * @see IComputerAccess#mount(String, IMount) |      * @see IComputerAccess#mount(String, IMount) | ||||||
|      * @see IComputerAccess#mountWritable(String, IWritableMount) |      * @see IComputerAccess#mountWritable(String, IWritableMount) | ||||||
|      * @see IMount |      * @see IMount | ||||||
|      */ |      */ | ||||||
|     @Nullable |     @Nullable | ||||||
|     public static IMount createResourceMount( @Nonnull Class<?> modClass, @Nonnull String domain, @Nonnull String subPath ) |     public static IMount createResourceMount( @Nonnull String domain, @Nonnull String subPath ) | ||||||
|     { |     { | ||||||
|         findCC(); |         return getInstance().createResourceMount( domain, subPath ); | ||||||
|         if( computerCraft_createResourceMount != null ) |     } | ||||||
|         { |  | ||||||
|             try |     /** | ||||||
|             { |      * Creates a file system mount to a resource folder, and returns it. | ||||||
|                 return (IMount) computerCraft_createResourceMount.invoke( null, modClass, domain, subPath ); |      * | ||||||
|             } |      * Use in conjunction with {@link IComputerAccess#mount} or {@link IComputerAccess#mountWritable} to mount a | ||||||
|             catch( Exception e ) |      * resource folder onto a computer's file system. | ||||||
|             { |      * | ||||||
|                 // It failed |      * The files in this mount will be a combination of files in all mod jar, and data packs that contain | ||||||
|             } |      * resources with the same domain and path. | ||||||
|         } |      * | ||||||
|         return null; |      * @param klass   The mod class to which the files belong. | ||||||
|  |      * @param domain  The domain under which to look for resources. eg: "mymod". | ||||||
|  |      * @param subPath The subPath under which to look for resources. eg: "lua/myfiles". | ||||||
|  |      * @return The mount, or {@code null} if it could be created for some reason. | ||||||
|  |      * @see IComputerAccess#mount(String, IMount) | ||||||
|  |      * @see IComputerAccess#mountWritable(String, IWritableMount) | ||||||
|  |      * @see IMount | ||||||
|  |      * @deprecated Use {@link #createResourceMount(String, String)} instead. | ||||||
|  |      */ | ||||||
|  |     @Nullable | ||||||
|  |     @Deprecated | ||||||
|  |     public static IMount createResourceMount( Class<?> klass, @Nonnull String domain, @Nonnull String subPath ) | ||||||
|  |     { | ||||||
|  |         return getInstance().createResourceMount( domain, subPath ); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
| @@ -178,18 +145,7 @@ public final class ComputerCraftAPI | |||||||
|      */ |      */ | ||||||
|     public static void registerPeripheralProvider( @Nonnull IPeripheralProvider provider ) |     public static void registerPeripheralProvider( @Nonnull IPeripheralProvider provider ) | ||||||
|     { |     { | ||||||
|         findCC(); |         getInstance().registerPeripheralProvider( provider ); | ||||||
|         if( computerCraft_registerPeripheralProvider != null ) |  | ||||||
|         { |  | ||||||
|             try |  | ||||||
|             { |  | ||||||
|                 computerCraft_registerPeripheralProvider.invoke( null, provider ); |  | ||||||
|             } |  | ||||||
|             catch( Exception e ) |  | ||||||
|             { |  | ||||||
|                 // It failed |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
| @@ -202,21 +158,7 @@ public final class ComputerCraftAPI | |||||||
|      */ |      */ | ||||||
|     public static void registerTurtleUpgrade( @Nonnull ITurtleUpgrade upgrade ) |     public static void registerTurtleUpgrade( @Nonnull ITurtleUpgrade upgrade ) | ||||||
|     { |     { | ||||||
|         if( upgrade != null ) |         getInstance().registerTurtleUpgrade( upgrade ); | ||||||
|         { |  | ||||||
|             findCC(); |  | ||||||
|             if( computerCraft_registerTurtleUpgrade != null ) |  | ||||||
|             { |  | ||||||
|                 try |  | ||||||
|                 { |  | ||||||
|                     computerCraft_registerTurtleUpgrade.invoke( null, upgrade ); |  | ||||||
|                 } |  | ||||||
|                 catch( Exception e ) |  | ||||||
|                 { |  | ||||||
|                     // It failed |  | ||||||
|                 } |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
| @@ -227,18 +169,7 @@ public final class ComputerCraftAPI | |||||||
|      */ |      */ | ||||||
|     public static void registerBundledRedstoneProvider( @Nonnull IBundledRedstoneProvider provider ) |     public static void registerBundledRedstoneProvider( @Nonnull IBundledRedstoneProvider provider ) | ||||||
|     { |     { | ||||||
|         findCC(); |         getInstance().registerBundledRedstoneProvider( provider ); | ||||||
|         if( computerCraft_registerBundledRedstoneProvider != null ) |  | ||||||
|         { |  | ||||||
|             try |  | ||||||
|             { |  | ||||||
|                 computerCraft_registerBundledRedstoneProvider.invoke( null, provider ); |  | ||||||
|             } |  | ||||||
|             catch( Exception e ) |  | ||||||
|             { |  | ||||||
|                 // It failed |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
| @@ -253,19 +184,7 @@ public final class ComputerCraftAPI | |||||||
|      */ |      */ | ||||||
|     public static int getBundledRedstoneOutput( @Nonnull World world, @Nonnull BlockPos pos, @Nonnull EnumFacing side ) |     public static int getBundledRedstoneOutput( @Nonnull World world, @Nonnull BlockPos pos, @Nonnull EnumFacing side ) | ||||||
|     { |     { | ||||||
|         findCC(); |         return getInstance().getBundledRedstoneOutput( world, pos, side ); | ||||||
|         if( computerCraft_getDefaultBundledRedstoneOutput != null ) |  | ||||||
|         { |  | ||||||
|             try |  | ||||||
|             { |  | ||||||
|                 return (Integer) computerCraft_getDefaultBundledRedstoneOutput.invoke( null, world, pos, side ); |  | ||||||
|             } |  | ||||||
|             catch( Exception e ) |  | ||||||
|             { |  | ||||||
|                 // It failed |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
|         return -1; |  | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
| @@ -276,58 +195,12 @@ public final class ComputerCraftAPI | |||||||
|      */ |      */ | ||||||
|     public static void registerMediaProvider( @Nonnull IMediaProvider provider ) |     public static void registerMediaProvider( @Nonnull IMediaProvider provider ) | ||||||
|     { |     { | ||||||
|         findCC(); |         getInstance().registerMediaProvider( provider ); | ||||||
|         if( computerCraft_registerMediaProvider != null ) |  | ||||||
|         { |  | ||||||
|             try |  | ||||||
|             { |  | ||||||
|                 computerCraft_registerMediaProvider.invoke( null, provider ); |  | ||||||
|             } |  | ||||||
|             catch( Exception e ) |  | ||||||
|             { |  | ||||||
|                 // It failed |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Registers a permission provider to restrict where turtles can move or build. |  | ||||||
|      * |  | ||||||
|      * @param provider The turtle permission provider to register. |  | ||||||
|      * @see ITurtlePermissionProvider |  | ||||||
|      * @deprecated Prefer using {@link dan200.computercraft.api.turtle.event.TurtleBlockEvent} or the standard Forge events. |  | ||||||
|      */ |  | ||||||
|     @Deprecated |  | ||||||
|     public static void registerPermissionProvider( @Nonnull ITurtlePermissionProvider provider ) |  | ||||||
|     { |  | ||||||
|         findCC(); |  | ||||||
|         if( computerCraft_registerPermissionProvider != null ) |  | ||||||
|         { |  | ||||||
|             try |  | ||||||
|             { |  | ||||||
|                 computerCraft_registerPermissionProvider.invoke( null, provider ); |  | ||||||
|             } |  | ||||||
|             catch( Exception e ) |  | ||||||
|             { |  | ||||||
|                 // It failed |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     public static void registerPocketUpgrade( @Nonnull IPocketUpgrade upgrade ) |     public static void registerPocketUpgrade( @Nonnull IPocketUpgrade upgrade ) | ||||||
|     { |     { | ||||||
|         findCC(); |         getInstance().registerPocketUpgrade( upgrade ); | ||||||
|         if( computerCraft_registerPocketUpgrade != null ) |  | ||||||
|         { |  | ||||||
|             try |  | ||||||
|             { |  | ||||||
|                 computerCraft_registerPocketUpgrade.invoke( null, upgrade ); |  | ||||||
|             } |  | ||||||
|             catch( Exception e ) |  | ||||||
|             { |  | ||||||
|                 // It failed |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
| @@ -337,36 +210,12 @@ public final class ComputerCraftAPI | |||||||
|      */ |      */ | ||||||
|     public static IPacketNetwork getWirelessNetwork() |     public static IPacketNetwork getWirelessNetwork() | ||||||
|     { |     { | ||||||
|         findCC(); |         return getInstance().getWirelessNetwork(); | ||||||
|         if( computerCraft_getWirelessNetwork != null ) |  | ||||||
|         { |  | ||||||
|             try |  | ||||||
|             { |  | ||||||
|                 return (IPacketNetwork) computerCraft_getWirelessNetwork.invoke( null ); |  | ||||||
|             } |  | ||||||
|             catch( Exception e ) |  | ||||||
|             { |  | ||||||
|                 // It failed; |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         return null; |  | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     public static void registerAPIFactory( @Nonnull ILuaAPIFactory upgrade ) |     public static void registerAPIFactory( @Nonnull ILuaAPIFactory factory ) | ||||||
|     { |     { | ||||||
|         findCC(); |         getInstance().registerAPIFactory( factory ); | ||||||
|         if( computerCraft_registerAPIFactory != null ) |  | ||||||
|         { |  | ||||||
|             try |  | ||||||
|             { |  | ||||||
|                 computerCraft_registerAPIFactory.invoke( null, upgrade ); |  | ||||||
|             } |  | ||||||
|             catch( Exception e ) |  | ||||||
|             { |  | ||||||
|                 // It failed |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
| @@ -379,22 +228,7 @@ public final class ComputerCraftAPI | |||||||
|     @Nonnull |     @Nonnull | ||||||
|     public static IWiredNode createWiredNodeForElement( @Nonnull IWiredElement element ) |     public static IWiredNode createWiredNodeForElement( @Nonnull IWiredElement element ) | ||||||
|     { |     { | ||||||
|         findCC(); |         return getInstance().createWiredNodeForElement( element ); | ||||||
|         if( computerCraft_createWiredNodeForElement != null ) |  | ||||||
|         { |  | ||||||
|             try |  | ||||||
|             { |  | ||||||
|                 return (IWiredNode) computerCraft_createWiredNodeForElement.invoke( null, element ); |  | ||||||
|             } |  | ||||||
|             catch( ReflectiveOperationException e ) |  | ||||||
|             { |  | ||||||
|                 throw new IllegalStateException( "Error creating wired node", e ); |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
|         else |  | ||||||
|         { |  | ||||||
|             throw new IllegalStateException( "ComputerCraft cannot be found" ); |  | ||||||
|         } |  | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
| @@ -407,117 +241,57 @@ public final class ComputerCraftAPI | |||||||
|      * @see IWiredElement#getNode() |      * @see IWiredElement#getNode() | ||||||
|      */ |      */ | ||||||
|     @Nullable |     @Nullable | ||||||
|     public static IWiredElement getWiredElementAt( @Nonnull IBlockAccess world, @Nonnull BlockPos pos, @Nonnull EnumFacing side ) |     public static IWiredElement getWiredElementAt( @Nonnull IBlockReader world, @Nonnull BlockPos pos, @Nonnull EnumFacing side ) | ||||||
|     { |     { | ||||||
|         findCC(); |         return getInstance().getWiredElementAt( world, pos, side ); | ||||||
|         if( computerCraft_getWiredElementAt != null ) |  | ||||||
|         { |  | ||||||
|             try |  | ||||||
|             { |  | ||||||
|                 return (IWiredElement) computerCraft_getWiredElementAt.invoke( null, world, pos, side ); |  | ||||||
|             } |  | ||||||
|             catch( ReflectiveOperationException ignored ) |  | ||||||
|             { |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         return null; |  | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     // The functions below here are private, and are used to interface with the non-API ComputerCraft classes. |     private static IComputerCraftAPI instance; | ||||||
|     // Reflection is used here so you can develop your mod without decompiling ComputerCraft and including |  | ||||||
|     // it in your solution, and so your mod won't crash if ComputerCraft is installed. |  | ||||||
|  |  | ||||||
|     private static void findCC() |     @Nonnull | ||||||
|  |     private static IComputerCraftAPI getInstance() | ||||||
|     { |     { | ||||||
|         if( !ccSearched ) |         if( instance != null ) return instance; | ||||||
|         { |  | ||||||
|             try |  | ||||||
|             { |  | ||||||
|                 computerCraft = Class.forName( "dan200.computercraft.ComputerCraft" ); |  | ||||||
|                 computerCraft_getVersion = findCCMethod( "getVersion", new Class<?>[] { |  | ||||||
|                 } ); |  | ||||||
|                 computerCraft_createUniqueNumberedSaveDir = findCCMethod( "createUniqueNumberedSaveDir", new Class<?>[] { |  | ||||||
|                     World.class, String.class |  | ||||||
|                 } ); |  | ||||||
|                 computerCraft_createSaveDirMount = findCCMethod( "createSaveDirMount", new Class<?>[] { |  | ||||||
|                     World.class, String.class, Long.TYPE |  | ||||||
|                 } ); |  | ||||||
|                 computerCraft_createResourceMount = findCCMethod( "createResourceMount", new Class<?>[] { |  | ||||||
|                     Class.class, String.class, String.class |  | ||||||
|                 } ); |  | ||||||
|                 computerCraft_registerPeripheralProvider = findCCMethod( "registerPeripheralProvider", new Class<?>[] { |  | ||||||
|                     IPeripheralProvider.class |  | ||||||
|                 } ); |  | ||||||
|                 computerCraft_registerTurtleUpgrade = findCCMethod( "registerTurtleUpgrade", new Class<?>[] { |  | ||||||
|                     ITurtleUpgrade.class |  | ||||||
|                 } ); |  | ||||||
|                 computerCraft_registerBundledRedstoneProvider = findCCMethod( "registerBundledRedstoneProvider", new Class<?>[] { |  | ||||||
|                     IBundledRedstoneProvider.class |  | ||||||
|                 } ); |  | ||||||
|                 computerCraft_getDefaultBundledRedstoneOutput = findCCMethod( "getDefaultBundledRedstoneOutput", new Class<?>[] { |  | ||||||
|                     World.class, BlockPos.class, EnumFacing.class |  | ||||||
|                 } ); |  | ||||||
|                 computerCraft_registerMediaProvider = findCCMethod( "registerMediaProvider", new Class<?>[] { |  | ||||||
|                     IMediaProvider.class |  | ||||||
|                 } ); |  | ||||||
|                 computerCraft_registerPermissionProvider = findCCMethod( "registerPermissionProvider", new Class<?>[] { |  | ||||||
|                     ITurtlePermissionProvider.class |  | ||||||
|                 } ); |  | ||||||
|                 computerCraft_registerPocketUpgrade = findCCMethod( "registerPocketUpgrade", new Class<?>[] { |  | ||||||
|                     IPocketUpgrade.class |  | ||||||
|                 } ); |  | ||||||
|                 computerCraft_getWirelessNetwork = findCCMethod( "getWirelessNetwork", new Class<?>[] { |  | ||||||
|                 } ); |  | ||||||
|                 computerCraft_registerAPIFactory = findCCMethod( "registerAPIFactory", new Class<?>[] { |  | ||||||
|                     ILuaAPIFactory.class |  | ||||||
|                 } ); |  | ||||||
|                 computerCraft_createWiredNodeForElement = findCCMethod( "createWiredNodeForElement", new Class<?>[] { |  | ||||||
|                     IWiredElement.class |  | ||||||
|                 } ); |  | ||||||
|                 computerCraft_getWiredElementAt = findCCMethod( "getWiredElementAt", new Class<?>[] { |  | ||||||
|                     IBlockAccess.class, BlockPos.class, EnumFacing.class |  | ||||||
|                 } ); |  | ||||||
|             } |  | ||||||
|             catch( Exception e ) |  | ||||||
|             { |  | ||||||
|                 System.err.println( "ComputerCraftAPI: ComputerCraft not found." ); |  | ||||||
|             } |  | ||||||
|             finally |  | ||||||
|             { |  | ||||||
|                 ccSearched = true; |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     private static Method findCCMethod( String name, Class<?>[] args ) |  | ||||||
|     { |  | ||||||
|         try |         try | ||||||
|         { |         { | ||||||
|             return computerCraft != null ? computerCraft.getMethod( name, args ) : null; |             return instance = (IComputerCraftAPI) Class.forName( "dan200.computercraft.ComputerCraftAPIImpl" ) | ||||||
|  |                 .getField( "INSTANCE" ).get( null ); | ||||||
|         } |         } | ||||||
|         catch( NoSuchMethodException e ) |         catch( ReflectiveOperationException e ) | ||||||
|         { |         { | ||||||
|             System.err.println( "ComputerCraftAPI: ComputerCraft method " + name + " not found." ); |             throw new IllegalStateException( "Cannot find ComputerCraft API", e ); | ||||||
|             return null; |  | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     private static boolean ccSearched = false; |     public interface IComputerCraftAPI | ||||||
|     private static Class<?> computerCraft = null; |     { | ||||||
|     private static Method computerCraft_getVersion = null; |         String getInstalledVersion(); | ||||||
|     private static Method computerCraft_createUniqueNumberedSaveDir = null; |  | ||||||
|     private static Method computerCraft_createSaveDirMount = null; |         int createUniqueNumberedSaveDir( @Nonnull World world, @Nonnull String parentSubPath ); | ||||||
|     private static Method computerCraft_createResourceMount = null; |  | ||||||
|     private static Method computerCraft_registerPeripheralProvider = null; |         IWritableMount createSaveDirMount( @Nonnull World world, @Nonnull String subPath, long capacity ); | ||||||
|     private static Method computerCraft_registerTurtleUpgrade = null; |  | ||||||
|     private static Method computerCraft_registerBundledRedstoneProvider = null; |         IMount createResourceMount( @Nonnull String domain, @Nonnull String subPath ); | ||||||
|     private static Method computerCraft_getDefaultBundledRedstoneOutput = null; |  | ||||||
|     private static Method computerCraft_registerMediaProvider = null; |         void registerPeripheralProvider( @Nonnull IPeripheralProvider provider ); | ||||||
|     private static Method computerCraft_registerPermissionProvider = null; |  | ||||||
|     private static Method computerCraft_registerPocketUpgrade = null; |         void registerTurtleUpgrade( @Nonnull ITurtleUpgrade upgrade ); | ||||||
|     private static Method computerCraft_getWirelessNetwork = null; |  | ||||||
|     private static Method computerCraft_registerAPIFactory = null; |         void registerBundledRedstoneProvider( @Nonnull IBundledRedstoneProvider provider ); | ||||||
|     private static Method computerCraft_createWiredNodeForElement = null; |  | ||||||
|     private static Method computerCraft_getWiredElementAt = null; |         int getBundledRedstoneOutput( @Nonnull World world, @Nonnull BlockPos pos, @Nonnull EnumFacing side ); | ||||||
|  |  | ||||||
|  |         void registerMediaProvider( @Nonnull IMediaProvider provider ); | ||||||
|  |  | ||||||
|  |         void registerPocketUpgrade( @Nonnull IPocketUpgrade upgrade ); | ||||||
|  |  | ||||||
|  |         IPacketNetwork getWirelessNetwork(); | ||||||
|  |  | ||||||
|  |         void registerAPIFactory( @Nonnull ILuaAPIFactory factory ); | ||||||
|  |  | ||||||
|  |         IWiredNode createWiredNodeForElement( @Nonnull IWiredElement element ); | ||||||
|  |  | ||||||
|  |         IWiredElement getWiredElementAt( @Nonnull IBlockReader world, @Nonnull BlockPos pos, @Nonnull EnumFacing side ); | ||||||
|  |     } | ||||||
| } | } | ||||||
|   | |||||||
| @@ -1,10 +0,0 @@ | |||||||
| /* |  | ||||||
|  * This file is part of the public ComputerCraft API - http://www.computercraft.info |  | ||||||
|  * Copyright Daniel Ratcliffe, 2011-2019. This API may be redistributed unmodified and in full only. |  | ||||||
|  * For help using the API, and posting your mods, visit the forums at computercraft.info. |  | ||||||
|  */ |  | ||||||
|  |  | ||||||
| @API( owner = "ComputerCraft", provides = "ComputerCraft|API|FileSystem", apiVersion = "${version}" ) |  | ||||||
| package dan200.computercraft.api.filesystem; |  | ||||||
|  |  | ||||||
| import net.minecraftforge.fml.common.API; |  | ||||||
| @@ -1,10 +0,0 @@ | |||||||
| /* |  | ||||||
|  * This file is part of the public ComputerCraft API - http://www.computercraft.info |  | ||||||
|  * Copyright Daniel Ratcliffe, 2011-2019. This API may be redistributed unmodified and in full only. |  | ||||||
|  * For help using the API, and posting your mods, visit the forums at computercraft.info. |  | ||||||
|  */ |  | ||||||
|  |  | ||||||
| @API( owner = "ComputerCraft", provides = "ComputerCraft|API|Lua", apiVersion = "${version}" ) |  | ||||||
| package dan200.computercraft.api.lua; |  | ||||||
|  |  | ||||||
| import net.minecraftforge.fml.common.API; |  | ||||||
| @@ -1,10 +0,0 @@ | |||||||
| /* |  | ||||||
|  * This file is part of the public ComputerCraft API - http://www.computercraft.info |  | ||||||
|  * Copyright Daniel Ratcliffe, 2011-2019. This API may be redistributed unmodified and in full only. |  | ||||||
|  * For help using the API, and posting your mods, visit the forums at computercraft.info. |  | ||||||
|  */ |  | ||||||
|  |  | ||||||
| @API( owner = "ComputerCraft", provides = "ComputerCraft|API|Media", apiVersion = "${version}" ) |  | ||||||
| package dan200.computercraft.api.media; |  | ||||||
|  |  | ||||||
| import net.minecraftforge.fml.common.API; |  | ||||||
| @@ -1,10 +0,0 @@ | |||||||
| /* |  | ||||||
|  * This file is part of the public ComputerCraft API - http://www.computercraft.info |  | ||||||
|  * Copyright Daniel Ratcliffe, 2011-2019. This API may be redistributed unmodified and in full only. |  | ||||||
|  * For help using the API, and posting your mods, visit the forums at computercraft.info. |  | ||||||
|  */ |  | ||||||
|  |  | ||||||
| @API( owner = "ComputerCraft", provides = "ComputerCraft|API|Network", apiVersion = "${version}" ) |  | ||||||
| package dan200.computercraft.api.network; |  | ||||||
|  |  | ||||||
| import net.minecraftforge.fml.common.API; |  | ||||||
| @@ -1,10 +0,0 @@ | |||||||
| /* |  | ||||||
|  * This file is part of the public ComputerCraft API - http://www.computercraft.info |  | ||||||
|  * Copyright Daniel Ratcliffe, 2011-2019. This API may be redistributed unmodified and in full only. |  | ||||||
|  * For help using the API, and posting your mods, visit the forums at computercraft.info. |  | ||||||
|  */ |  | ||||||
|  |  | ||||||
| @API( owner = "ComputerCraft", provides = "ComputerCraft|API|Network|Wired", apiVersion = "${version}" ) |  | ||||||
| package dan200.computercraft.api.network.wired; |  | ||||||
|  |  | ||||||
| import net.minecraftforge.fml.common.API; |  | ||||||
| @@ -1,10 +0,0 @@ | |||||||
| /* |  | ||||||
|  * This file is part of the public ComputerCraft API - http://www.computercraft.info |  | ||||||
|  * Copyright Daniel Ratcliffe, 2011-2019. This API may be redistributed unmodified and in full only. |  | ||||||
|  * For help using the API, and posting your mods, visit the forums at computercraft.info. |  | ||||||
|  */ |  | ||||||
|  |  | ||||||
| @API( owner = "ComputerCraft", provides = "ComputerCraft|API", apiVersion = "${version}" ) |  | ||||||
| package dan200.computercraft.api; |  | ||||||
|  |  | ||||||
| import net.minecraftforge.fml.common.API; |  | ||||||
| @@ -1,10 +0,0 @@ | |||||||
| /* |  | ||||||
|  * This file is part of the public ComputerCraft API - http://www.computercraft.info |  | ||||||
|  * Copyright Daniel Ratcliffe, 2011-2019. This API may be redistributed unmodified and in full only. |  | ||||||
|  * For help using the API, and posting your mods, visit the forums at computercraft.info. |  | ||||||
|  */ |  | ||||||
|  |  | ||||||
| @API( owner = "ComputerCraft", provides = "ComputerCraft|API|Peripheral", apiVersion = "${version}" ) |  | ||||||
| package dan200.computercraft.api.peripheral; |  | ||||||
|  |  | ||||||
| import net.minecraftforge.fml.common.API; |  | ||||||
| @@ -1,42 +0,0 @@ | |||||||
| /* |  | ||||||
|  * This file is part of the public ComputerCraft API - http://www.computercraft.info |  | ||||||
|  * Copyright Daniel Ratcliffe, 2011-2019. This API may be redistributed unmodified and in full only. |  | ||||||
|  * For help using the API, and posting your mods, visit the forums at computercraft.info. |  | ||||||
|  */ |  | ||||||
|  |  | ||||||
| package dan200.computercraft.api.permissions; |  | ||||||
|  |  | ||||||
| import net.minecraft.util.math.BlockPos; |  | ||||||
| import net.minecraft.world.World; |  | ||||||
|  |  | ||||||
| import javax.annotation.Nonnull; |  | ||||||
|  |  | ||||||
| /** |  | ||||||
|  * This interface is used to restrict where turtles can move or build. |  | ||||||
|  * |  | ||||||
|  * Turtles will call these methods before attempting to perform an action, allowing them to be cancelled. |  | ||||||
|  * |  | ||||||
|  * @see dan200.computercraft.api.ComputerCraftAPI#registerPermissionProvider(ITurtlePermissionProvider) |  | ||||||
|  */ |  | ||||||
| public interface ITurtlePermissionProvider |  | ||||||
| { |  | ||||||
|     /** |  | ||||||
|      * Determine whether a block can be entered by a turtle. |  | ||||||
|      * |  | ||||||
|      * @param world The world the block exists in |  | ||||||
|      * @param pos   The location of the block. |  | ||||||
|      * @return Whether the turtle can move into this block. |  | ||||||
|      */ |  | ||||||
|     boolean isBlockEnterable( @Nonnull World world, @Nonnull BlockPos pos ); |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Determine whether a block can be modified by a turtle. |  | ||||||
|      * |  | ||||||
|      * This includes breaking and placing blocks. |  | ||||||
|      * |  | ||||||
|      * @param world The world the block exists in |  | ||||||
|      * @param pos   The location of the block. |  | ||||||
|      * @return Whether the turtle can modify this block. |  | ||||||
|      */ |  | ||||||
|     boolean isBlockEditable( @Nonnull World world, @Nonnull BlockPos pos ); |  | ||||||
| } |  | ||||||
| @@ -1,10 +0,0 @@ | |||||||
| /* |  | ||||||
|  * This file is part of the public ComputerCraft API - http://www.computercraft.info |  | ||||||
|  * Copyright Daniel Ratcliffe, 2011-2019. This API may be redistributed unmodified and in full only. |  | ||||||
|  * For help using the API, and posting your mods, visit the forums at computercraft.info. |  | ||||||
|  */ |  | ||||||
|  |  | ||||||
| @API( owner = "ComputerCraft", provides = "ComputerCraft|API|Permissions", apiVersion = "${version}" ) |  | ||||||
| package dan200.computercraft.api.permissions; |  | ||||||
|  |  | ||||||
| import net.minecraftforge.fml.common.API; |  | ||||||
| @@ -7,7 +7,9 @@ | |||||||
| package dan200.computercraft.api.pocket; | package dan200.computercraft.api.pocket; | ||||||
|  |  | ||||||
| import net.minecraft.item.ItemStack; | import net.minecraft.item.ItemStack; | ||||||
|  | import net.minecraft.util.IItemProvider; | ||||||
| import net.minecraft.util.ResourceLocation; | import net.minecraft.util.ResourceLocation; | ||||||
|  | import net.minecraft.util.Util; | ||||||
|  |  | ||||||
| import javax.annotation.Nonnull; | import javax.annotation.Nonnull; | ||||||
|  |  | ||||||
| @@ -29,9 +31,14 @@ public abstract class AbstractPocketUpgrade implements IPocketUpgrade | |||||||
|         this.stack = stack; |         this.stack = stack; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     protected AbstractPocketUpgrade( ResourceLocation id, ItemStack stack ) |     protected AbstractPocketUpgrade( ResourceLocation id, String adjective, IItemProvider item ) | ||||||
|     { |     { | ||||||
|         this( id, "upgrade." + id + ".adjective", stack ); |         this( id, adjective, new ItemStack( item ) ); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     protected AbstractPocketUpgrade( ResourceLocation id, IItemProvider item ) | ||||||
|  |     { | ||||||
|  |         this( id, Util.makeTranslationKey( "upgrade", id ) + ".adjective", new ItemStack( item ) ); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @Nonnull |     @Nonnull | ||||||
|   | |||||||
| @@ -23,22 +23,12 @@ public interface IPocketAccess | |||||||
|     /** |     /** | ||||||
|      * Gets the entity holding this item. |      * Gets the entity holding this item. | ||||||
|      * |      * | ||||||
|      * @return The holding entity. This may be {@code null}. |  | ||||||
|      * @deprecated Use {@link #getValidEntity()} where possible. |  | ||||||
|      */ |  | ||||||
|     @Nullable |  | ||||||
|     @Deprecated |  | ||||||
|     Entity getEntity(); |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Gets the entity holding this item with additional safety checks. |  | ||||||
|      * |  | ||||||
|      * This must be called on the server thread. |      * This must be called on the server thread. | ||||||
|      * |      * | ||||||
|      * @return The holding entity, or {@code null} if none exists. |      * @return The holding entity, or {@code null} if none exists. | ||||||
|      */ |      */ | ||||||
|     @Nullable |     @Nullable | ||||||
|     Entity getValidEntity(); |     Entity getEntity(); | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * Get the colour of this pocket computer as a RGB number. |      * Get the colour of this pocket computer as a RGB number. | ||||||
|   | |||||||
| @@ -1,10 +0,0 @@ | |||||||
| /* |  | ||||||
|  * This file is part of the public ComputerCraft API - http://www.computercraft.info |  | ||||||
|  * Copyright Daniel Ratcliffe, 2011-2019. This API may be redistributed unmodified and in full only. |  | ||||||
|  * For help using the API, and posting your mods, visit the forums at computercraft.info. |  | ||||||
|  */ |  | ||||||
|  |  | ||||||
| @API( owner = "ComputerCraft", provides = "ComputerCraft|API|Redstone", apiVersion = "${version}" ) |  | ||||||
| package dan200.computercraft.api.redstone; |  | ||||||
|  |  | ||||||
| import net.minecraftforge.fml.common.API; |  | ||||||
| @@ -54,8 +54,7 @@ public interface ITurtleAccess | |||||||
|      * @param world The new world to move it to |      * @param world The new world to move it to | ||||||
|      * @param pos   The new position to move it to. |      * @param pos   The new position to move it to. | ||||||
|      * @return Whether the movement was successful. It may fail if the block was not loaded or the block placement |      * @return Whether the movement was successful. It may fail if the block was not loaded or the block placement | ||||||
|      * was cancelled. Note this will not check |      * was cancelled. | ||||||
|      * {@link dan200.computercraft.api.permissions.ITurtlePermissionProvider#isBlockEnterable(World, BlockPos)}. |  | ||||||
|      * @throws UnsupportedOperationException When attempting to teleport on the client side. |      * @throws UnsupportedOperationException When attempting to teleport on the client side. | ||||||
|      */ |      */ | ||||||
|     boolean teleportTo( @Nonnull World world, @Nonnull BlockPos pos ); |     boolean teleportTo( @Nonnull World world, @Nonnull BlockPos pos ); | ||||||
|   | |||||||
| @@ -10,15 +10,15 @@ import dan200.computercraft.api.ComputerCraftAPI; | |||||||
| import dan200.computercraft.api.peripheral.IPeripheral; | import dan200.computercraft.api.peripheral.IPeripheral; | ||||||
| import dan200.computercraft.api.turtle.event.TurtleAttackEvent; | import dan200.computercraft.api.turtle.event.TurtleAttackEvent; | ||||||
| import dan200.computercraft.api.turtle.event.TurtleBlockEvent; | import dan200.computercraft.api.turtle.event.TurtleBlockEvent; | ||||||
| import net.minecraft.client.renderer.block.model.IBakedModel; | import net.minecraft.client.renderer.model.IBakedModel; | ||||||
| import net.minecraft.client.renderer.block.model.ModelResourceLocation; | import net.minecraft.client.renderer.model.ModelResourceLocation; | ||||||
| import net.minecraft.item.ItemStack; | import net.minecraft.item.ItemStack; | ||||||
| import net.minecraft.util.EnumFacing; | import net.minecraft.util.EnumFacing; | ||||||
| import net.minecraft.util.ResourceLocation; | import net.minecraft.util.ResourceLocation; | ||||||
|  | import net.minecraftforge.api.distmarker.Dist; | ||||||
|  | import net.minecraftforge.api.distmarker.OnlyIn; | ||||||
| import net.minecraftforge.event.entity.player.AttackEntityEvent; | import net.minecraftforge.event.entity.player.AttackEntityEvent; | ||||||
| import net.minecraftforge.event.world.BlockEvent; | import net.minecraftforge.event.world.BlockEvent; | ||||||
| import net.minecraftforge.fml.relauncher.Side; |  | ||||||
| import net.minecraftforge.fml.relauncher.SideOnly; |  | ||||||
| import org.apache.commons.lang3.tuple.Pair; | import org.apache.commons.lang3.tuple.Pair; | ||||||
|  |  | ||||||
| import javax.annotation.Nonnull; | import javax.annotation.Nonnull; | ||||||
| @@ -44,17 +44,6 @@ public interface ITurtleUpgrade | |||||||
|     @Nonnull |     @Nonnull | ||||||
|     ResourceLocation getUpgradeID(); |     ResourceLocation getUpgradeID(); | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Gets a numerical identifier representing this type of turtle upgrade, |  | ||||||
|      * for backwards compatibility with pre-1.76 worlds. If your upgrade was |  | ||||||
|      * not released for older ComputerCraft versions, you can return -1 here. |  | ||||||
|      * The turtle will fail registration if an already used positive ID is specified. |  | ||||||
|      * |  | ||||||
|      * @return The legacy ID, or -1 if is needed. |  | ||||||
|      * @see ComputerCraftAPI#registerTurtleUpgrade(ITurtleUpgrade) |  | ||||||
|      */ |  | ||||||
|     int getLegacyUpgradeID(); |  | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * Return an unlocalised string to describe this type of turtle in turtle item names. |      * Return an unlocalised string to describe this type of turtle in turtle item names. | ||||||
|      * |      * | ||||||
| @@ -133,7 +122,7 @@ public interface ITurtleUpgrade | |||||||
|      * Called to obtain the model to be used when rendering a turtle peripheral. |      * Called to obtain the model to be used when rendering a turtle peripheral. | ||||||
|      * |      * | ||||||
|      * This can be obtained from {@link net.minecraft.client.renderer.ItemModelMesher#getItemModel(ItemStack)}, |      * This can be obtained from {@link net.minecraft.client.renderer.ItemModelMesher#getItemModel(ItemStack)}, | ||||||
|      * {@link net.minecraft.client.renderer.block.model.ModelManager#getModel(ModelResourceLocation)} or any other |      * {@link net.minecraft.client.renderer.model.ModelManager#getModel(ModelResourceLocation)} or any other | ||||||
|      * source. |      * source. | ||||||
|      * |      * | ||||||
|      * @param turtle Access to the turtle that the upgrade resides on. This will be null when getting item models! |      * @param turtle Access to the turtle that the upgrade resides on. This will be null when getting item models! | ||||||
| @@ -142,7 +131,7 @@ public interface ITurtleUpgrade | |||||||
|      * a transformation of {@code null} has the same effect as the identify matrix. |      * a transformation of {@code null} has the same effect as the identify matrix. | ||||||
|      */ |      */ | ||||||
|     @Nonnull |     @Nonnull | ||||||
|     @SideOnly( Side.CLIENT ) |     @OnlyIn( Dist.CLIENT ) | ||||||
|     Pair<IBakedModel, Matrix4f> getModel( @Nullable ITurtleAccess turtle, @Nonnull TurtleSide side ); |     Pair<IBakedModel, Matrix4f> getModel( @Nullable ITurtleAccess turtle, @Nonnull TurtleSide side ); | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|   | |||||||
| @@ -8,7 +8,7 @@ package dan200.computercraft.api.turtle.event; | |||||||
|  |  | ||||||
| import dan200.computercraft.api.turtle.ITurtleAccess; | import dan200.computercraft.api.turtle.ITurtleAccess; | ||||||
| import dan200.computercraft.api.turtle.TurtleCommandResult; | import dan200.computercraft.api.turtle.TurtleCommandResult; | ||||||
| import net.minecraftforge.fml.common.eventhandler.Cancelable; | import net.minecraftforge.eventbus.api.Cancelable; | ||||||
|  |  | ||||||
| import javax.annotation.Nonnull; | import javax.annotation.Nonnull; | ||||||
| import javax.annotation.Nullable; | import javax.annotation.Nullable; | ||||||
|   | |||||||
| @@ -7,7 +7,7 @@ | |||||||
| package dan200.computercraft.api.turtle.event; | package dan200.computercraft.api.turtle.event; | ||||||
|  |  | ||||||
| import dan200.computercraft.api.turtle.ITurtleAccess; | import dan200.computercraft.api.turtle.ITurtleAccess; | ||||||
| import net.minecraftforge.fml.common.eventhandler.Event; | import net.minecraftforge.eventbus.api.Event; | ||||||
|  |  | ||||||
| import javax.annotation.Nonnull; | import javax.annotation.Nonnull; | ||||||
| import java.util.Objects; | import java.util.Objects; | ||||||
|   | |||||||
| @@ -1,10 +0,0 @@ | |||||||
| /* |  | ||||||
|  * This file is part of the public ComputerCraft API - http://www.computercraft.info |  | ||||||
|  * Copyright Daniel Ratcliffe, 2011-2019. This API may be redistributed unmodified and in full only. |  | ||||||
|  * For help using the API, and posting your mods, visit the forums at computercraft.info. |  | ||||||
|  */ |  | ||||||
|  |  | ||||||
| @API( owner = "ComputerCraft", provides = "ComputerCraft|API|Turtle|Event", apiVersion = "${version}" ) |  | ||||||
| package dan200.computercraft.api.turtle.event; |  | ||||||
|  |  | ||||||
| import net.minecraftforge.fml.common.API; |  | ||||||
| @@ -1,10 +0,0 @@ | |||||||
| /* |  | ||||||
|  * This file is part of the public ComputerCraft API - http://www.computercraft.info |  | ||||||
|  * Copyright Daniel Ratcliffe, 2011-2019. This API may be redistributed unmodified and in full only. |  | ||||||
|  * For help using the API, and posting your mods, visit the forums at computercraft.info. |  | ||||||
|  */ |  | ||||||
|  |  | ||||||
| @API( owner = "ComputerCraft", provides = "ComputerCraft|API|Turtle", apiVersion = "${version}" ) |  | ||||||
| package dan200.computercraft.api.turtle; |  | ||||||
|  |  | ||||||
| import net.minecraftforge.fml.common.API; |  | ||||||
| @@ -8,104 +8,81 @@ package dan200.computercraft.client; | |||||||
|  |  | ||||||
| import dan200.computercraft.ComputerCraft; | import dan200.computercraft.ComputerCraft; | ||||||
| import dan200.computercraft.client.render.TurtleModelLoader; | import dan200.computercraft.client.render.TurtleModelLoader; | ||||||
| import dan200.computercraft.shared.media.items.ItemDiskLegacy; | import dan200.computercraft.shared.common.IColouredItem; | ||||||
|  | import dan200.computercraft.shared.media.items.ItemDisk; | ||||||
| import dan200.computercraft.shared.pocket.items.ItemPocketComputer; | import dan200.computercraft.shared.pocket.items.ItemPocketComputer; | ||||||
| import dan200.computercraft.shared.turtle.items.ItemTurtleBase; |  | ||||||
| import dan200.computercraft.shared.util.Colour; | import dan200.computercraft.shared.util.Colour; | ||||||
| import net.minecraft.client.Minecraft; | import net.minecraft.client.Minecraft; | ||||||
| import net.minecraft.client.renderer.ItemMeshDefinition; | import net.minecraft.client.renderer.model.IBakedModel; | ||||||
| import net.minecraft.client.renderer.block.model.IBakedModel; | import net.minecraft.client.renderer.model.IUnbakedModel; | ||||||
| import net.minecraft.client.renderer.block.model.ModelBakery; | import net.minecraft.client.renderer.model.ModelResourceLocation; | ||||||
| import net.minecraft.client.renderer.block.model.ModelResourceLocation; | import net.minecraft.client.renderer.model.ModelRotation; | ||||||
| import net.minecraft.client.renderer.texture.TextureMap; |  | ||||||
| import net.minecraft.client.renderer.vertex.DefaultVertexFormats; | import net.minecraft.client.renderer.vertex.DefaultVertexFormats; | ||||||
| import net.minecraft.item.Item; | import net.minecraft.resources.IResourceManager; | ||||||
| import net.minecraft.item.ItemStack; |  | ||||||
| import net.minecraft.util.ResourceLocation; | import net.minecraft.util.ResourceLocation; | ||||||
|  | import net.minecraftforge.api.distmarker.Dist; | ||||||
| import net.minecraftforge.client.event.ColorHandlerEvent; | import net.minecraftforge.client.event.ColorHandlerEvent; | ||||||
| import net.minecraftforge.client.event.ModelBakeEvent; | import net.minecraftforge.client.event.ModelBakeEvent; | ||||||
| import net.minecraftforge.client.event.ModelRegistryEvent; | import net.minecraftforge.client.event.ModelRegistryEvent; | ||||||
| import net.minecraftforge.client.event.TextureStitchEvent; | import net.minecraftforge.client.event.TextureStitchEvent; | ||||||
| import net.minecraftforge.client.model.IModel; |  | ||||||
| import net.minecraftforge.client.model.ModelLoader; | import net.minecraftforge.client.model.ModelLoader; | ||||||
| import net.minecraftforge.client.model.ModelLoaderRegistry; | import net.minecraftforge.client.model.ModelLoaderRegistry; | ||||||
|  | import net.minecraftforge.eventbus.api.SubscribeEvent; | ||||||
| import net.minecraftforge.fml.common.Mod; | import net.minecraftforge.fml.common.Mod; | ||||||
| import net.minecraftforge.fml.common.eventhandler.SubscribeEvent; |  | ||||||
| import net.minecraftforge.fml.relauncher.Side; |  | ||||||
|  |  | ||||||
| import javax.annotation.Nonnull; | import java.util.HashSet; | ||||||
|  | import java.util.Map; | ||||||
|  |  | ||||||
| /** | /** | ||||||
|  * Registers textures and models for items. |  * Registers textures and models for items. | ||||||
|  */ |  */ | ||||||
| @Mod.EventBusSubscriber( modid = ComputerCraft.MOD_ID, value = Side.CLIENT ) | @Mod.EventBusSubscriber( modid = ComputerCraft.MOD_ID, value = Dist.CLIENT ) | ||||||
| public final class ClientRegistry | public final class ClientRegistry | ||||||
| { | { | ||||||
|     private static final String[] EXTRA_MODELS = new String[] { |     private static final String[] EXTRA_MODELS = new String[] { | ||||||
|         "turtle_modem_off_left", |         "turtle_modem_normal_off_left", | ||||||
|         "turtle_modem_on_left", |         "turtle_modem_normal_on_left", | ||||||
|         "turtle_modem_off_right", |         "turtle_modem_normal_off_right", | ||||||
|         "turtle_modem_on_right", |         "turtle_modem_normal_on_right", | ||||||
|  |  | ||||||
|  |         "turtle_modem_advanced_off_left", | ||||||
|  |         "turtle_modem_advanced_on_left", | ||||||
|  |         "turtle_modem_advanced_off_right", | ||||||
|  |         "turtle_modem_advanced_on_right", | ||||||
|         "turtle_crafting_table_left", |         "turtle_crafting_table_left", | ||||||
|         "turtle_crafting_table_right", |         "turtle_crafting_table_right", | ||||||
|         "advanced_turtle_modem_off_left", |  | ||||||
|         "advanced_turtle_modem_on_left", |  | ||||||
|         "advanced_turtle_modem_off_right", |  | ||||||
|         "advanced_turtle_modem_on_right", |  | ||||||
|         "turtle_speaker_upgrade_left", |         "turtle_speaker_upgrade_left", | ||||||
|         "turtle_speaker_upgrade_right", |         "turtle_speaker_upgrade_right", | ||||||
|  |  | ||||||
|         "turtle_white", |         "turtle_colour", | ||||||
|         "turtle_elf_overlay", |         "turtle_elf_overlay", | ||||||
|     }; |     }; | ||||||
|  |  | ||||||
|  |     private static final String[] EXTRA_TEXTURES = new String[] { | ||||||
|  |         // TODO: Gather these automatically from the model. I'm unable to get this working with Forge's current | ||||||
|  |         //  model loading code. | ||||||
|  |         "block/turtle_colour", | ||||||
|  |         "block/turtle_elf_overlay", | ||||||
|  |         "block/turtle_crafty_face", | ||||||
|  |         "block/turtle_speaker_face", | ||||||
|  |     }; | ||||||
|  |  | ||||||
|     private ClientRegistry() {} |     private ClientRegistry() {} | ||||||
|  |  | ||||||
|     @SubscribeEvent |     @SubscribeEvent | ||||||
|     public static void registerModels( ModelRegistryEvent event ) |     public static void registerModels( ModelRegistryEvent event ) | ||||||
|     { |     { | ||||||
|         ModelLoaderRegistry.registerLoader( TurtleModelLoader.INSTANCE ); |         ModelLoaderRegistry.registerLoader( TurtleModelLoader.INSTANCE ); | ||||||
|  |  | ||||||
|         // Register item models |  | ||||||
|         registerUniversalItemModel( ComputerCraft.Items.computer, "computer" ); |  | ||||||
|         registerItemModel( ComputerCraft.Items.commandComputer, 0, "command_computer" ); |  | ||||||
|  |  | ||||||
|         registerItemModel( ComputerCraft.Items.pocketComputer, 0, "pocket_computer" ); |  | ||||||
|         registerItemModel( ComputerCraft.Items.pocketComputer, 1, "advanced_pocket_computer" ); |  | ||||||
|  |  | ||||||
|         registerItemModel( ComputerCraft.Items.peripheral, 0, "peripheral" ); |  | ||||||
|         registerItemModel( ComputerCraft.Items.peripheral, 1, "wireless_modem" ); |  | ||||||
|         registerItemModel( ComputerCraft.Items.peripheral, 2, "monitor" ); |  | ||||||
|         registerItemModel( ComputerCraft.Items.peripheral, 3, "printer" ); |  | ||||||
|         registerItemModel( ComputerCraft.Items.peripheral, 4, "advanced_monitor" ); |  | ||||||
|         registerItemModel( ComputerCraft.Items.cable, 0, "cable" ); |  | ||||||
|         registerItemModel( ComputerCraft.Items.cable, 1, "wired_modem" ); |  | ||||||
|         registerItemModel( ComputerCraft.Items.advancedModem, 0, "advanced_modem" ); |  | ||||||
|         registerItemModel( ComputerCraft.Items.peripheral, 5, "speaker" ); |  | ||||||
|         registerItemModel( ComputerCraft.Items.wiredModemFull, 0, "wired_modem_full" ); |  | ||||||
|  |  | ||||||
|         registerUniversalItemModel( ComputerCraft.Items.disk, "disk" ); |  | ||||||
|         registerItemModel( ComputerCraft.Items.diskExpanded, 0, "disk_expanded" ); |  | ||||||
|         registerItemModel( ComputerCraft.Items.treasureDisk, 0, "treasure_disk" ); |  | ||||||
|  |  | ||||||
|         registerItemModel( ComputerCraft.Items.printout, 0, "printout" ); |  | ||||||
|         registerItemModel( ComputerCraft.Items.printout, 1, "pages" ); |  | ||||||
|         registerItemModel( ComputerCraft.Items.printout, 2, "book" ); |  | ||||||
|  |  | ||||||
|         registerUniversalItemModel( ComputerCraft.Items.turtle, "turtle" ); |  | ||||||
|         registerUniversalItemModel( ComputerCraft.Items.turtleExpanded, "turtle" ); |  | ||||||
|         registerUniversalItemModel( ComputerCraft.Items.turtleAdvanced, "turtle_advanced" ); |  | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @SubscribeEvent |     @SubscribeEvent | ||||||
|     public static void onTextureStitchEvent( TextureStitchEvent.Pre event ) |     public static void onTextureStitchEvent( TextureStitchEvent.Pre event ) | ||||||
|     { |     { | ||||||
|         // Load all textures for the extra models |         IResourceManager manager = Minecraft.getInstance().getResourceManager(); | ||||||
|         TextureMap map = event.getMap(); |         for( String extra : EXTRA_TEXTURES ) | ||||||
|         for( String upgrade : EXTRA_MODELS ) |  | ||||||
|         { |         { | ||||||
|             IModel model = ModelLoaderRegistry.getModelOrMissing( new ResourceLocation( "computercraft", "block/" + upgrade ) ); |             event.getMap().registerSprite( manager, new ResourceLocation( ComputerCraft.MOD_ID, extra ) ); | ||||||
|             for( ResourceLocation texture : model.getTextures() ) map.registerSprite( texture ); |  | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  |  | ||||||
| @@ -113,73 +90,73 @@ public final class ClientRegistry | |||||||
|     public static void onModelBakeEvent( ModelBakeEvent event ) |     public static void onModelBakeEvent( ModelBakeEvent event ) | ||||||
|     { |     { | ||||||
|         // Load all extra models |         // Load all extra models | ||||||
|         for( String model : EXTRA_MODELS ) loadBlockModel( event, model ); |         ModelLoader loader = event.getModelLoader(); | ||||||
|  |         Map<ModelResourceLocation, IBakedModel> registry = event.getModelRegistry(); | ||||||
|  |  | ||||||
|  |         for( String model : EXTRA_MODELS ) | ||||||
|  |         { | ||||||
|  |             IBakedModel bakedModel = bake( loader, loader.getUnbakedModel( new ResourceLocation( ComputerCraft.MOD_ID, "item/" + model ) ) ); | ||||||
|  |  | ||||||
|  |             if( bakedModel != null ) | ||||||
|  |             { | ||||||
|  |                 registry.put( | ||||||
|  |                     new ModelResourceLocation( new ResourceLocation( ComputerCraft.MOD_ID, model ), "inventory" ), | ||||||
|  |                     bakedModel | ||||||
|  |                 ); | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         // And load the custom turtle models in too. | ||||||
|  |         registry.put( | ||||||
|  |             new ModelResourceLocation( new ResourceLocation( ComputerCraft.MOD_ID, "turtle_normal" ), "inventory" ), | ||||||
|  |             bake( loader, TurtleModelLoader.INSTANCE.loadModel( new ResourceLocation( ComputerCraft.MOD_ID, "item/turtle_normal" ) ) ) | ||||||
|  |         ); | ||||||
|  |  | ||||||
|  |         registry.put( | ||||||
|  |             new ModelResourceLocation( new ResourceLocation( ComputerCraft.MOD_ID, "turtle_advanced" ), "inventory" ), | ||||||
|  |             bake( loader, TurtleModelLoader.INSTANCE.loadModel( new ResourceLocation( ComputerCraft.MOD_ID, "item/turtle_advanced" ) ) ) | ||||||
|  |         ); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @SubscribeEvent |     @SubscribeEvent | ||||||
|     public static void onItemColours( ColorHandlerEvent.Item event ) |     public static void onItemColours( ColorHandlerEvent.Item event ) | ||||||
|     { |     { | ||||||
|         event.getItemColors().registerItemColorHandler( |         event.getItemColors().register( | ||||||
|             ( stack, layer ) -> layer == 1 ? ((ItemDiskLegacy) stack.getItem()).getColour( stack ) : 0xFFFFFF, |             ( stack, layer ) -> layer == 1 ? ((ItemDisk) stack.getItem()).getColour( stack ) : 0xFFFFFF, | ||||||
|             ComputerCraft.Items.disk, ComputerCraft.Items.diskExpanded |             ComputerCraft.Items.disk | ||||||
|         ); |         ); | ||||||
|  |  | ||||||
|         event.getItemColors().registerItemColorHandler( ( stack, layer ) -> { |         event.getItemColors().register( ( stack, layer ) -> { | ||||||
|             switch( layer ) |             switch( layer ) | ||||||
|             { |             { | ||||||
|                 case 0: |                 case 0: | ||||||
|                 default: |                 default: | ||||||
|                     return 0xFFFFFF; |                     return 0xFFFFFF; | ||||||
|                 case 1: // Frame colour |                 case 1: // Frame colour | ||||||
|                     return ComputerCraft.Items.pocketComputer.getColour( stack ); |                     return IColouredItem.getColourBasic( stack ); | ||||||
|                 case 2: // Light colour |                 case 2: // Light colour | ||||||
|                 { |                 { | ||||||
|                     int light = ItemPocketComputer.getLightState( stack ); |                     int light = ItemPocketComputer.getLightState( stack ); | ||||||
|                     return light == -1 ? Colour.Black.getHex() : light; |                     return light == -1 ? Colour.Black.getHex() : light; | ||||||
|                 } |                 } | ||||||
|             } |             } | ||||||
|         }, ComputerCraft.Items.pocketComputer ); |         }, ComputerCraft.Items.pocketComputerNormal, ComputerCraft.Items.pocketComputerAdvanced ); | ||||||
|  |  | ||||||
|         // Setup turtle colours |         // Setup turtle colours | ||||||
|         event.getItemColors().registerItemColorHandler( |         event.getItemColors().register( | ||||||
|             ( stack, tintIndex ) -> tintIndex == 0 ? ((ItemTurtleBase) stack.getItem()).getColour( stack ) : 0xFFFFFF, |             ( stack, tintIndex ) -> tintIndex == 0 ? ((IColouredItem) stack.getItem()).getColour( stack ) : 0xFFFFFF, | ||||||
|             ComputerCraft.Blocks.turtle, ComputerCraft.Blocks.turtleExpanded, ComputerCraft.Blocks.turtleAdvanced |             ComputerCraft.Blocks.turtleNormal, ComputerCraft.Blocks.turtleAdvanced | ||||||
|         ); |         ); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     private static void registerItemModel( Item item, int damage, String name ) |     private static IBakedModel bake( ModelLoader loader, IUnbakedModel model ) | ||||||
|     { |     { | ||||||
|         ResourceLocation location = new ResourceLocation( ComputerCraft.MOD_ID, name ); |         model.getTextures( loader::getUnbakedModel, new HashSet<>() ); | ||||||
|         final ModelResourceLocation res = new ModelResourceLocation( location, "inventory" ); |  | ||||||
|         ModelBakery.registerItemVariants( item, location ); |  | ||||||
|         ModelLoader.setCustomModelResourceLocation( item, damage, res ); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     private static void registerUniversalItemModel( Item item, String mainModel ) |         return model.bake( | ||||||
|     { |             loader::getUnbakedModel, | ||||||
|         ResourceLocation mainLocation = new ResourceLocation( ComputerCraft.MOD_ID, mainModel ); |             ModelLoader.defaultTextureGetter(), | ||||||
|         ModelBakery.registerItemVariants( item, mainLocation ); |             ModelRotation.X0_Y0, false, DefaultVertexFormats.BLOCK | ||||||
|  |  | ||||||
|         final ModelResourceLocation mainModelLocation = new ModelResourceLocation( mainLocation, "inventory" ); |  | ||||||
|         ModelLoader.setCustomMeshDefinition( item, new ItemMeshDefinition() |  | ||||||
|         { |  | ||||||
|             @Nonnull |  | ||||||
|             @Override |  | ||||||
|             public ModelResourceLocation getModelLocation( @Nonnull ItemStack stack ) |  | ||||||
|             { |  | ||||||
|                 return mainModelLocation; |  | ||||||
|             } |  | ||||||
|         } ); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     private static void loadBlockModel( ModelBakeEvent event, String name ) |  | ||||||
|     { |  | ||||||
|         IModel model = ModelLoaderRegistry.getModelOrMissing( new ResourceLocation( ComputerCraft.MOD_ID, "block/" + name ) ); |  | ||||||
|         IBakedModel bakedModel = model.bake( |  | ||||||
|             model.getDefaultState(), DefaultVertexFormats.ITEM, |  | ||||||
|             location -> Minecraft.getMinecraft().getTextureMapBlocks().getAtlasSprite( location.toString() ) |  | ||||||
|         ); |         ); | ||||||
|  |  | ||||||
|         event.getModelRegistry().putObject( new ModelResourceLocation( ComputerCraft.MOD_ID + ":" + name, "inventory" ), bakedModel ); |  | ||||||
|     } |     } | ||||||
| } | } | ||||||
|   | |||||||
| @@ -30,7 +30,7 @@ public class ClientTableFormatter implements TableFormatter | |||||||
|  |  | ||||||
|     private static FontRenderer renderer() |     private static FontRenderer renderer() | ||||||
|     { |     { | ||||||
|         return Minecraft.getMinecraft().fontRenderer; |         return Minecraft.getInstance().fontRenderer; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @Override |     @Override | ||||||
| @@ -42,7 +42,7 @@ public class ClientTableFormatter implements TableFormatter | |||||||
|  |  | ||||||
|         FontRenderer renderer = renderer(); |         FontRenderer renderer = renderer(); | ||||||
|  |  | ||||||
|         float spaceWidth = renderer.getCharWidth( ' ' ); |         float spaceWidth = renderer.getStringWidth( " " ); | ||||||
|         int spaces = MathHelper.floor( extraWidth / spaceWidth ); |         int spaces = MathHelper.floor( extraWidth / spaceWidth ); | ||||||
|         int extra = extraWidth - (int) (spaces * spaceWidth); |         int extra = extraWidth - (int) (spaces * spaceWidth); | ||||||
|  |  | ||||||
| @@ -64,11 +64,11 @@ public class ClientTableFormatter implements TableFormatter | |||||||
|     @Override |     @Override | ||||||
|     public void writeLine( int id, ITextComponent component ) |     public void writeLine( int id, ITextComponent component ) | ||||||
|     { |     { | ||||||
|         Minecraft mc = Minecraft.getMinecraft(); |         Minecraft mc = Minecraft.getInstance(); | ||||||
|         GuiNewChat chat = mc.ingameGUI.getChatGUI(); |         GuiNewChat chat = mc.ingameGUI.getChatGUI(); | ||||||
|  |  | ||||||
|         // Trim the text if it goes over the allowed length |         // Trim the text if it goes over the allowed length | ||||||
|         int maxWidth = MathHelper.floor( chat.getChatWidth() / chat.getChatScale() ); |         int maxWidth = MathHelper.floor( chat.getChatWidth() / chat.getScale() ); | ||||||
|         List<ITextComponent> list = GuiUtilRenderComponents.splitText( component, maxWidth, mc.fontRenderer, false, false ); |         List<ITextComponent> list = GuiUtilRenderComponents.splitText( component, maxWidth, mc.fontRenderer, false, false ); | ||||||
|         if( !list.isEmpty() ) chat.printChatMessageWithOptionalDeletion( list.get( 0 ), id ); |         if( !list.isEmpty() ) chat.printChatMessageWithOptionalDeletion( list.get( 0 ), id ); | ||||||
|     } |     } | ||||||
| @@ -76,7 +76,7 @@ public class ClientTableFormatter implements TableFormatter | |||||||
|     @Override |     @Override | ||||||
|     public int display( TableBuilder table ) |     public int display( TableBuilder table ) | ||||||
|     { |     { | ||||||
|         GuiNewChat chat = Minecraft.getMinecraft().ingameGUI.getChatGUI(); |         GuiNewChat chat = Minecraft.getInstance().ingameGUI.getChatGUI(); | ||||||
|  |  | ||||||
|         int lastHeight = lastHeights.get( table.getId() ); |         int lastHeight = lastHeights.get( table.getId() ); | ||||||
|  |  | ||||||
|   | |||||||
| @@ -7,12 +7,12 @@ | |||||||
| package dan200.computercraft.client; | package dan200.computercraft.client; | ||||||
|  |  | ||||||
| import dan200.computercraft.ComputerCraft; | import dan200.computercraft.ComputerCraft; | ||||||
|  | import net.minecraftforge.api.distmarker.Dist; | ||||||
|  | import net.minecraftforge.eventbus.api.SubscribeEvent; | ||||||
| import net.minecraftforge.fml.common.Mod; | import net.minecraftforge.fml.common.Mod; | ||||||
| import net.minecraftforge.fml.common.eventhandler.SubscribeEvent; |  | ||||||
| import net.minecraftforge.fml.common.gameevent.TickEvent; | import net.minecraftforge.fml.common.gameevent.TickEvent; | ||||||
| import net.minecraftforge.fml.relauncher.Side; |  | ||||||
|  |  | ||||||
| @Mod.EventBusSubscriber( modid = ComputerCraft.MOD_ID, value = Side.CLIENT ) | @Mod.EventBusSubscriber( modid = ComputerCraft.MOD_ID, value = Dist.CLIENT ) | ||||||
| public final class FrameInfo | public final class FrameInfo | ||||||
| { | { | ||||||
|     private static int tick; |     private static int tick; | ||||||
|   | |||||||
| @@ -39,7 +39,7 @@ public final class FixedWidthFontRenderer | |||||||
|  |  | ||||||
|     private FixedWidthFontRenderer() |     private FixedWidthFontRenderer() | ||||||
|     { |     { | ||||||
|         m_textureManager = Minecraft.getMinecraft().getTextureManager(); |         m_textureManager = Minecraft.getInstance().getTextureManager(); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     private static void greyscaleify( double[] rgb ) |     private static void greyscaleify( double[] rgb ) | ||||||
| @@ -195,6 +195,6 @@ public final class FixedWidthFontRenderer | |||||||
|     public void bindFont() |     public void bindFont() | ||||||
|     { |     { | ||||||
|         m_textureManager.bindTexture( FONT ); |         m_textureManager.bindTexture( FONT ); | ||||||
|         GlStateManager.glTexParameteri( GL11.GL_TEXTURE_2D, GL11.GL_TEXTURE_WRAP_S, GL11.GL_CLAMP ); |         GlStateManager.texParameteri( GL11.GL_TEXTURE_2D, GL11.GL_TEXTURE_WRAP_S, GL11.GL_CLAMP ); | ||||||
|     } |     } | ||||||
| } | } | ||||||
|   | |||||||
| @@ -8,23 +8,19 @@ package dan200.computercraft.client.gui; | |||||||
|  |  | ||||||
| import dan200.computercraft.ComputerCraft; | import dan200.computercraft.ComputerCraft; | ||||||
| import dan200.computercraft.client.gui.widgets.WidgetTerminal; | import dan200.computercraft.client.gui.widgets.WidgetTerminal; | ||||||
|  | import dan200.computercraft.client.gui.widgets.WidgetWrapper; | ||||||
| import dan200.computercraft.shared.computer.blocks.TileComputer; | import dan200.computercraft.shared.computer.blocks.TileComputer; | ||||||
| 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.computer.core.IComputer; |  | ||||||
| import dan200.computercraft.shared.computer.inventory.ContainerComputer; | import dan200.computercraft.shared.computer.inventory.ContainerComputer; | ||||||
| import net.minecraft.client.gui.inventory.GuiContainer; | import net.minecraft.client.gui.inventory.GuiContainer; | ||||||
| import net.minecraft.client.renderer.GlStateManager; | import net.minecraft.client.renderer.GlStateManager; | ||||||
| import net.minecraft.inventory.Container; | import net.minecraft.inventory.Container; | ||||||
| import net.minecraft.util.ResourceLocation; | import net.minecraft.util.ResourceLocation; | ||||||
| import org.lwjgl.input.Keyboard; |  | ||||||
| import org.lwjgl.input.Mouse; |  | ||||||
|  |  | ||||||
| import java.io.IOException; |  | ||||||
|  |  | ||||||
| public class GuiComputer extends GuiContainer | public class GuiComputer extends GuiContainer | ||||||
| { | { | ||||||
|     private static final ResourceLocation BACKGROUND = new ResourceLocation( "computercraft", "textures/gui/corners.png" ); |     private static final ResourceLocation BACKGROUND_NORMAL = new ResourceLocation( "computercraft", "textures/gui/corners_normal.png" ); | ||||||
|     private static final ResourceLocation BACKGROUND_ADVANCED = new ResourceLocation( "computercraft", "textures/gui/corners_advanced.png" ); |     private static final ResourceLocation BACKGROUND_ADVANCED = new ResourceLocation( "computercraft", "textures/gui/corners_advanced.png" ); | ||||||
|     private static final ResourceLocation BACKGROUND_COMMAND = new ResourceLocation( "computercraft", "textures/gui/corners_command.png" ); |     private static final ResourceLocation BACKGROUND_COMMAND = new ResourceLocation( "computercraft", "textures/gui/corners_command.png" ); | ||||||
|  |  | ||||||
| @@ -32,7 +28,9 @@ public class GuiComputer extends GuiContainer | |||||||
|     private final ClientComputer m_computer; |     private final ClientComputer m_computer; | ||||||
|     private final int m_termWidth; |     private final int m_termWidth; | ||||||
|     private final int m_termHeight; |     private final int m_termHeight; | ||||||
|     private WidgetTerminal m_terminal; |  | ||||||
|  |     private WidgetTerminal terminal; | ||||||
|  |     private WidgetWrapper terminalWrapper; | ||||||
|  |  | ||||||
|     public GuiComputer( Container container, ComputerFamily family, ClientComputer computer, int termWidth, int termHeight ) |     public GuiComputer( Container container, ComputerFamily family, ClientComputer computer, int termWidth, int termHeight ) | ||||||
|     { |     { | ||||||
| @@ -41,13 +39,7 @@ public class GuiComputer extends GuiContainer | |||||||
|         m_computer = computer; |         m_computer = computer; | ||||||
|         m_termWidth = termWidth; |         m_termWidth = termWidth; | ||||||
|         m_termHeight = termHeight; |         m_termHeight = termHeight; | ||||||
|         m_terminal = null; |         terminal = null; | ||||||
|     } |  | ||||||
|  |  | ||||||
|     @Deprecated |  | ||||||
|     public GuiComputer( Container container, ComputerFamily family, IComputer computer, int termWidth, int termHeight ) |  | ||||||
|     { |  | ||||||
|         this( container, family, (ClientComputer) computer, termWidth, termHeight ); |  | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     public GuiComputer( TileComputer computer ) |     public GuiComputer( TileComputer computer ) | ||||||
| @@ -62,103 +54,60 @@ public class GuiComputer extends GuiContainer | |||||||
|     } |     } | ||||||
|  |  | ||||||
|     @Override |     @Override | ||||||
|     public void initGui() |     protected void initGui() | ||||||
|     { |     { | ||||||
|         super.initGui(); |         mc.keyboardListener.enableRepeatEvents( true ); | ||||||
|         Keyboard.enableRepeatEvents( true ); |  | ||||||
|  |  | ||||||
|         m_terminal = new WidgetTerminal( 0, 0, m_termWidth, m_termHeight, () -> m_computer, 2, 2, 2, 2 ); |         int termPxWidth = m_termWidth * FixedWidthFontRenderer.FONT_WIDTH; | ||||||
|         m_terminal.setAllowFocusLoss( false ); |         int termPxHeight = m_termHeight * FixedWidthFontRenderer.FONT_HEIGHT; | ||||||
|         xSize = m_terminal.getWidth() + 24; |  | ||||||
|         ySize = m_terminal.getHeight() + 24; |         xSize = termPxWidth + 4 + 24; | ||||||
|  |         ySize = termPxHeight + 4 + 24; | ||||||
|  |  | ||||||
|  |         super.initGui(); | ||||||
|  |  | ||||||
|  |         terminal = new WidgetTerminal( mc, () -> m_computer, m_termWidth, m_termHeight, 2, 2, 2, 2 ); | ||||||
|  |         terminalWrapper = new WidgetWrapper( terminal, 2 + 12 + guiLeft, 2 + 12 + guiTop, termPxWidth, termPxHeight ); | ||||||
|  |  | ||||||
|  |         children.add( terminalWrapper ); | ||||||
|  |         setFocused( terminalWrapper ); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @Override |     @Override | ||||||
|     public void onGuiClosed() |     public void onGuiClosed() | ||||||
|     { |     { | ||||||
|         super.onGuiClosed(); |         super.onGuiClosed(); | ||||||
|         Keyboard.enableRepeatEvents( false ); |         children.remove( terminal ); | ||||||
|  |         terminal = null; | ||||||
|  |         mc.keyboardListener.enableRepeatEvents( false ); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @Override |     @Override | ||||||
|     public void updateScreen() |     public void tick() | ||||||
|     { |     { | ||||||
|         super.updateScreen(); |         super.tick(); | ||||||
|         m_terminal.update(); |         terminal.update(); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @Override |     @Override | ||||||
|     protected void keyTyped( char c, int k ) throws IOException |     public void drawGuiContainerBackgroundLayer( float partialTicks, int mouseX, int mouseY ) | ||||||
|     { |  | ||||||
|         if( k == 1 ) |  | ||||||
|         { |  | ||||||
|             super.keyTyped( c, k ); |  | ||||||
|         } |  | ||||||
|         else |  | ||||||
|         { |  | ||||||
|             if( m_terminal.onKeyTyped( c, k ) ) keyHandled = true; |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     @Override |  | ||||||
|     protected void mouseClicked( int x, int y, int button ) |  | ||||||
|     { |  | ||||||
|         int startX = (width - m_terminal.getWidth()) / 2; |  | ||||||
|         int startY = (height - m_terminal.getHeight()) / 2; |  | ||||||
|         m_terminal.mouseClicked( x - startX, y - startY, button ); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     @Override |  | ||||||
|     public void handleMouseInput() throws IOException |  | ||||||
|     { |  | ||||||
|         super.handleMouseInput(); |  | ||||||
|  |  | ||||||
|         int x = Mouse.getEventX() * width / mc.displayWidth; |  | ||||||
|         int y = height - Mouse.getEventY() * height / mc.displayHeight - 1; |  | ||||||
|         int startX = (width - m_terminal.getWidth()) / 2; |  | ||||||
|         int startY = (height - m_terminal.getHeight()) / 2; |  | ||||||
|         m_terminal.handleMouseInput( x - startX, y - startY ); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     @Override |  | ||||||
|     public void handleKeyboardInput() throws IOException |  | ||||||
|     { |  | ||||||
|         super.handleKeyboardInput(); |  | ||||||
|         if( m_terminal.onKeyboardInput() ) keyHandled = true; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     @Override |  | ||||||
|     protected void drawGuiContainerForegroundLayer( int par1, int par2 ) |  | ||||||
|     { |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     @Override |  | ||||||
|     protected void drawGuiContainerBackgroundLayer( float var1, int var2, int var3 ) |  | ||||||
|     { |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     @Override |  | ||||||
|     public void drawScreen( int mouseX, int mouseY, float partialTicks ) |  | ||||||
|     { |     { | ||||||
|         // Work out where to draw |         // Work out where to draw | ||||||
|         int startX = (width - m_terminal.getWidth()) / 2; |         int startX = terminalWrapper.getX() - 2; | ||||||
|         int startY = (height - m_terminal.getHeight()) / 2; |         int startY = terminalWrapper.getY() - 2; | ||||||
|         int endX = startX + m_terminal.getWidth(); |         int endX = startX + terminalWrapper.getWidth() + 4; | ||||||
|         int endY = startY + m_terminal.getHeight(); |         int endY = startY + terminalWrapper.getHeight() + 4; | ||||||
|  |  | ||||||
|         // Draw background |  | ||||||
|         drawDefaultBackground(); |  | ||||||
|  |  | ||||||
|         // Draw terminal |         // Draw terminal | ||||||
|         m_terminal.draw( mc, startX, startY, mouseX, mouseY ); |         terminal.draw( terminalWrapper.getX(), terminalWrapper.getY() ); | ||||||
|  |  | ||||||
|         // Draw a border around the terminal |         // Draw a border around the terminal | ||||||
|         GlStateManager.color( 1.0f, 1.0f, 1.0f, 1.0f ); |         GlStateManager.color4f( 1.0f, 1.0f, 1.0f, 1.0f ); | ||||||
|         switch( m_family ) |         switch( m_family ) | ||||||
|         { |         { | ||||||
|             case Normal: |             case Normal: | ||||||
|             default: |             default: | ||||||
|                 mc.getTextureManager().bindTexture( BACKGROUND ); |                 mc.getTextureManager().bindTexture( BACKGROUND_NORMAL ); | ||||||
|                 break; |                 break; | ||||||
|             case Advanced: |             case Advanced: | ||||||
|                 mc.getTextureManager().bindTexture( BACKGROUND_ADVANCED ); |                 mc.getTextureManager().bindTexture( BACKGROUND_ADVANCED ); | ||||||
| @@ -179,4 +128,26 @@ public class GuiComputer extends GuiContainer | |||||||
|         drawTexturedModalRect( startX - 12, startY, 0, 28, 12, endY - startY ); |         drawTexturedModalRect( startX - 12, startY, 0, 28, 12, endY - startY ); | ||||||
|         drawTexturedModalRect( endX, startY, 36, 28, 12, endY - startY ); |         drawTexturedModalRect( endX, startY, 36, 28, 12, endY - startY ); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     @Override | ||||||
|  |     public void render( int mouseX, int mouseY, float partialTicks ) | ||||||
|  |     { | ||||||
|  |         drawDefaultBackground(); | ||||||
|  |         super.render( mouseX, mouseY, partialTicks ); | ||||||
|  |         renderHoveredToolTip( mouseX, mouseY ); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     @Override | ||||||
|  |     public boolean mouseDragged( double x, double y, int button, double deltaX, double deltaY ) | ||||||
|  |     { | ||||||
|  |         return (getFocused() != null && getFocused().mouseDragged( x, y, button, deltaX, deltaY )) | ||||||
|  |             || super.mouseDragged( x, y, button, deltaX, deltaY ); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     @Override | ||||||
|  |     public boolean mouseReleased( double x, double y, int button ) | ||||||
|  |     { | ||||||
|  |         return (getFocused() != null && getFocused().mouseReleased( x, y, button )) | ||||||
|  |             || super.mouseReleased( x, y, button ); | ||||||
|  |     } | ||||||
| } | } | ||||||
|   | |||||||
| @@ -1,50 +0,0 @@ | |||||||
| /* |  | ||||||
|  * This file is part of ComputerCraft - http://www.computercraft.info |  | ||||||
|  * Copyright Daniel Ratcliffe, 2011-2019. Do not distribute without permission. |  | ||||||
|  * Send enquiries to dratcliffe@gmail.com |  | ||||||
|  */ |  | ||||||
|  |  | ||||||
| package dan200.computercraft.client.gui; |  | ||||||
|  |  | ||||||
| import dan200.computercraft.ComputerCraft; |  | ||||||
| import dan200.computercraft.shared.Config; |  | ||||||
| import net.minecraft.client.Minecraft; |  | ||||||
| import net.minecraft.client.gui.GuiScreen; |  | ||||||
| import net.minecraftforge.fml.client.IModGuiFactory; |  | ||||||
| import net.minecraftforge.fml.client.config.GuiConfig; |  | ||||||
|  |  | ||||||
| import java.util.Set; |  | ||||||
|  |  | ||||||
| public class GuiConfigCC extends GuiConfig |  | ||||||
| { |  | ||||||
|     public GuiConfigCC( GuiScreen parentScreen ) |  | ||||||
|     { |  | ||||||
|         super( parentScreen, Config.getConfigElements(), ComputerCraft.MOD_ID, false, false, "CC: Tweaked" ); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     public static class Factory implements IModGuiFactory |  | ||||||
|     { |  | ||||||
|         @Override |  | ||||||
|         public void initialize( Minecraft minecraft ) |  | ||||||
|         { |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         @Override |  | ||||||
|         public boolean hasConfigGui() |  | ||||||
|         { |  | ||||||
|             return true; |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         @Override |  | ||||||
|         public GuiScreen createConfigGui( GuiScreen parentScreen ) |  | ||||||
|         { |  | ||||||
|             return new GuiConfigCC( parentScreen ); |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         @Override |  | ||||||
|         public Set<RuntimeOptionCategoryElement> runtimeGuiCategories() |  | ||||||
|         { |  | ||||||
|             return null; |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| @@ -14,7 +14,7 @@ import net.minecraft.util.ResourceLocation; | |||||||
|  |  | ||||||
| public class GuiDiskDrive extends GuiContainer | public class GuiDiskDrive extends GuiContainer | ||||||
| { | { | ||||||
|     private static final ResourceLocation BACKGROUND = new ResourceLocation( "computercraft", "textures/gui/diskdrive.png" ); |     private static final ResourceLocation BACKGROUND = new ResourceLocation( "computercraft", "textures/gui/disk_drive.png" ); | ||||||
|  |  | ||||||
|     private final ContainerDiskDrive m_container; |     private final ContainerDiskDrive m_container; | ||||||
|  |  | ||||||
| @@ -27,24 +27,24 @@ public class GuiDiskDrive extends GuiContainer | |||||||
|     @Override |     @Override | ||||||
|     protected void drawGuiContainerForegroundLayer( int mouseX, int mouseY ) |     protected void drawGuiContainerForegroundLayer( int mouseX, int mouseY ) | ||||||
|     { |     { | ||||||
|         String title = m_container.getDiskDrive().getDisplayName().getUnformattedText(); |         String title = m_container.getDiskDrive().getDisplayName().getString(); | ||||||
|         fontRenderer.drawString( title, (xSize - fontRenderer.getStringWidth( title )) / 2, 6, 0x404040 ); |         fontRenderer.drawString( title, (xSize - fontRenderer.getStringWidth( title )) / 2.0f, 6, 0x404040 ); | ||||||
|         fontRenderer.drawString( I18n.format( "container.inventory" ), 8, ySize - 96 + 2, 0x404040 ); |         fontRenderer.drawString( I18n.format( "container.inventory" ), 8, ySize - 96 + 2, 0x404040 ); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @Override |     @Override | ||||||
|     protected void drawGuiContainerBackgroundLayer( float partialTicks, int mouseX, int mouseY ) |     protected void drawGuiContainerBackgroundLayer( float partialTicks, int mouseX, int mouseY ) | ||||||
|     { |     { | ||||||
|         GlStateManager.color( 1.0F, 1.0F, 1.0F, 1.0F ); |         GlStateManager.color4f( 1.0F, 1.0F, 1.0F, 1.0F ); | ||||||
|         mc.getTextureManager().bindTexture( BACKGROUND ); |         mc.getTextureManager().bindTexture( BACKGROUND ); | ||||||
|         drawTexturedModalRect( guiLeft, guiTop, 0, 0, xSize, ySize ); |         drawTexturedModalRect( guiLeft, guiTop, 0, 0, xSize, ySize ); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @Override |     @Override | ||||||
|     public void drawScreen( int mouseX, int mouseY, float partialTicks ) |     public void render( int mouseX, int mouseY, float partialTicks ) | ||||||
|     { |     { | ||||||
|         drawDefaultBackground(); |         drawDefaultBackground(); | ||||||
|         super.drawScreen( mouseX, mouseY, partialTicks ); |         super.render( mouseX, mouseY, partialTicks ); | ||||||
|         renderHoveredToolTip( mouseX, mouseY ); |         renderHoveredToolTip( mouseX, mouseY ); | ||||||
|     } |     } | ||||||
| } | } | ||||||
|   | |||||||
| @@ -7,19 +7,28 @@ | |||||||
| package dan200.computercraft.client.gui; | package dan200.computercraft.client.gui; | ||||||
|  |  | ||||||
| import dan200.computercraft.ComputerCraft; | import dan200.computercraft.ComputerCraft; | ||||||
| import dan200.computercraft.shared.media.inventory.ContainerHeldItem; | import dan200.computercraft.shared.computer.core.ComputerFamily; | ||||||
|  | import dan200.computercraft.shared.pocket.inventory.ContainerPocketComputer; | ||||||
| import dan200.computercraft.shared.pocket.items.ItemPocketComputer; | import dan200.computercraft.shared.pocket.items.ItemPocketComputer; | ||||||
|  | import net.minecraft.item.Item; | ||||||
|  | import net.minecraft.item.ItemStack; | ||||||
|  |  | ||||||
| public class GuiPocketComputer extends GuiComputer | public class GuiPocketComputer extends GuiComputer | ||||||
| { | { | ||||||
|     public GuiPocketComputer( ContainerHeldItem container ) |     public GuiPocketComputer( ContainerPocketComputer container ) | ||||||
|     { |     { | ||||||
|         super( |         super( | ||||||
|             container, |             container, | ||||||
|             ComputerCraft.Items.pocketComputer.getFamily( container.getStack() ), |             getFamily( container.getStack() ), | ||||||
|             ItemPocketComputer.createClientComputer( container.getStack() ), |             ItemPocketComputer.createClientComputer( container.getStack() ), | ||||||
|             ComputerCraft.terminalWidth_pocketComputer, |             ComputerCraft.terminalWidth_pocketComputer, | ||||||
|             ComputerCraft.terminalHeight_pocketComputer |             ComputerCraft.terminalHeight_pocketComputer | ||||||
|         ); |         ); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     private static ComputerFamily getFamily( ItemStack stack ) | ||||||
|  |     { | ||||||
|  |         Item item = stack.getItem(); | ||||||
|  |         return item instanceof ItemPocketComputer ? ((ItemPocketComputer) item).getFamily() : ComputerFamily.Normal; | ||||||
|  |     } | ||||||
| } | } | ||||||
|   | |||||||
| @@ -27,15 +27,15 @@ public class GuiPrinter extends GuiContainer | |||||||
|     @Override |     @Override | ||||||
|     protected void drawGuiContainerForegroundLayer( int mouseX, int mouseY ) |     protected void drawGuiContainerForegroundLayer( int mouseX, int mouseY ) | ||||||
|     { |     { | ||||||
|         String title = container.getPrinter().getDisplayName().getUnformattedText(); |         String title = container.getPrinter().getDisplayName().getString(); | ||||||
|         fontRenderer.drawString( title, (xSize - fontRenderer.getStringWidth( title )) / 2, 6, 0x404040 ); |         fontRenderer.drawString( title, (xSize - fontRenderer.getStringWidth( title )) / 2.0f, 6, 0x404040 ); | ||||||
|         fontRenderer.drawString( I18n.format( "container.inventory" ), 8, ySize - 96 + 2, 0x404040 ); |         fontRenderer.drawString( I18n.format( "container.inventory" ), 8, ySize - 96 + 2, 0x404040 ); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @Override |     @Override | ||||||
|     protected void drawGuiContainerBackgroundLayer( float partialTicks, int mouseX, int mouseY ) |     protected void drawGuiContainerBackgroundLayer( float partialTicks, int mouseX, int mouseY ) | ||||||
|     { |     { | ||||||
|         GlStateManager.color( 1.0F, 1.0F, 1.0F, 1.0F ); |         GlStateManager.color4f( 1.0F, 1.0F, 1.0F, 1.0F ); | ||||||
|         mc.getTextureManager().bindTexture( BACKGROUND ); |         mc.getTextureManager().bindTexture( BACKGROUND ); | ||||||
|         drawTexturedModalRect( guiLeft, guiTop, 0, 0, xSize, ySize ); |         drawTexturedModalRect( guiLeft, guiTop, 0, 0, xSize, ySize ); | ||||||
|  |  | ||||||
| @@ -43,10 +43,10 @@ public class GuiPrinter extends GuiContainer | |||||||
|     } |     } | ||||||
|  |  | ||||||
|     @Override |     @Override | ||||||
|     public void drawScreen( int mouseX, int mouseY, float partialTicks ) |     public void render( int mouseX, int mouseY, float partialTicks ) | ||||||
|     { |     { | ||||||
|         drawDefaultBackground(); |         drawDefaultBackground(); | ||||||
|         super.drawScreen( mouseX, mouseY, partialTicks ); |         super.render( mouseX, mouseY, partialTicks ); | ||||||
|         renderHoveredToolTip( mouseX, mouseY ); |         renderHoveredToolTip( mouseX, mouseY ); | ||||||
|     } |     } | ||||||
| } | } | ||||||
|   | |||||||
| @@ -7,13 +7,11 @@ | |||||||
| package dan200.computercraft.client.gui; | package dan200.computercraft.client.gui; | ||||||
|  |  | ||||||
| import dan200.computercraft.core.terminal.TextBuffer; | import dan200.computercraft.core.terminal.TextBuffer; | ||||||
| import dan200.computercraft.shared.media.inventory.ContainerHeldItem; | import dan200.computercraft.shared.common.ContainerHeldItem; | ||||||
| import dan200.computercraft.shared.media.items.ItemPrintout; | import dan200.computercraft.shared.media.items.ItemPrintout; | ||||||
| import net.minecraft.client.gui.inventory.GuiContainer; | import net.minecraft.client.gui.inventory.GuiContainer; | ||||||
| import net.minecraft.client.renderer.GlStateManager; | import net.minecraft.client.renderer.GlStateManager; | ||||||
| import org.lwjgl.input.Mouse; | import org.lwjgl.glfw.GLFW; | ||||||
|  |  | ||||||
| import java.io.IOException; |  | ||||||
|  |  | ||||||
| import static dan200.computercraft.client.render.PrintoutRenderer.*; | import static dan200.computercraft.client.render.PrintoutRenderer.*; | ||||||
|  |  | ||||||
| @@ -41,63 +39,70 @@ public class GuiPrintout extends GuiContainer | |||||||
|  |  | ||||||
|         m_page = 0; |         m_page = 0; | ||||||
|         m_pages = Math.max( m_text.length / ItemPrintout.LINES_PER_PAGE, 1 ); |         m_pages = Math.max( m_text.length / ItemPrintout.LINES_PER_PAGE, 1 ); | ||||||
|         m_book = ItemPrintout.getType( container.getStack() ) == ItemPrintout.Type.Book; |         m_book = ((ItemPrintout) container.getStack().getItem()).getType() == ItemPrintout.Type.BOOK; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @Override |     @Override | ||||||
|     protected void keyTyped( char c, int k ) throws IOException |     public boolean keyPressed( int key, int scancode, int modifiers ) | ||||||
|     { |     { | ||||||
|         super.keyTyped( c, k ); |         if( super.keyPressed( key, scancode, modifiers ) ) return true; | ||||||
|  |  | ||||||
|         if( k == 205 ) |         if( key == GLFW.GLFW_KEY_RIGHT ) | ||||||
|         { |         { | ||||||
|             // Right |  | ||||||
|             if( m_page < m_pages - 1 ) m_page++; |             if( m_page < m_pages - 1 ) m_page++; | ||||||
|  |             return true; | ||||||
|         } |         } | ||||||
|         else if( k == 203 ) |  | ||||||
|  |         if( key == GLFW.GLFW_KEY_LEFT ) | ||||||
|         { |         { | ||||||
|             // Left |  | ||||||
|             if( m_page > 0 ) m_page--; |             if( m_page > 0 ) m_page--; | ||||||
|  |             return true; | ||||||
|         } |         } | ||||||
|  |  | ||||||
|  |         return false; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @Override |     @Override | ||||||
|     public void handleMouseInput() throws IOException |     public boolean mouseScrolled( double delta ) | ||||||
|     { |     { | ||||||
|         super.handleMouseInput(); |         if( super.mouseScrolled( delta ) ) return true; | ||||||
|  |         if( delta < 0 ) | ||||||
|         int mouseWheelChange = Mouse.getEventDWheel(); |  | ||||||
|         if( mouseWheelChange < 0 ) |  | ||||||
|         { |         { | ||||||
|             // Scroll up goes to the next page |             // Scroll up goes to the next page | ||||||
|             if( m_page < m_pages - 1 ) m_page++; |             if( m_page < m_pages - 1 ) m_page++; | ||||||
|  |             return true; | ||||||
|         } |         } | ||||||
|         else if( mouseWheelChange > 0 ) |  | ||||||
|  |         if( delta > 0 ) | ||||||
|         { |         { | ||||||
|             // Scroll down goes to the previous page |             // Scroll down goes to the previous page | ||||||
|             if( m_page > 0 ) m_page--; |             if( m_page > 0 ) m_page--; | ||||||
|  |             return true; | ||||||
|         } |         } | ||||||
|  |  | ||||||
|  |         return false; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @Override |     @Override | ||||||
|     protected void drawGuiContainerBackgroundLayer( float partialTicks, int mouseX, int mouseY ) |     public void drawGuiContainerBackgroundLayer( float partialTicks, int mouseX, int mouseY ) | ||||||
|     { |     { | ||||||
|         // Draw the printout |         // Draw the printout | ||||||
|         GlStateManager.color( 1.0f, 1.0f, 1.0f, 1.0f ); |         GlStateManager.color4f( 1.0f, 1.0f, 1.0f, 1.0f ); | ||||||
|  |         GlStateManager.enableDepthTest(); | ||||||
|  |  | ||||||
|         drawBorder( guiLeft, guiTop, zLevel, m_page, m_pages, m_book ); |         drawBorder( guiLeft, guiTop, zLevel, m_page, m_pages, m_book ); | ||||||
|         drawText( guiLeft + X_TEXT_MARGIN, guiTop + Y_TEXT_MARGIN, ItemPrintout.LINES_PER_PAGE * m_page, m_text, m_colours ); |         drawText( guiLeft + X_TEXT_MARGIN, guiTop + Y_TEXT_MARGIN, ItemPrintout.LINES_PER_PAGE * m_page, m_text, m_colours ); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @Override |     @Override | ||||||
|     public void drawScreen( int mouseX, int mouseY, float partialTicks ) |     public void render( int mouseX, int mouseY, float partialTicks ) | ||||||
|     { |     { | ||||||
|         // We must take the background further back in order to not overlap with our printed pages. |         // We must take the background further back in order to not overlap with our printed pages. | ||||||
|         zLevel--; |         zLevel--; | ||||||
|         drawDefaultBackground(); |         drawDefaultBackground(); | ||||||
|         zLevel++; |         zLevel++; | ||||||
|  |  | ||||||
|         super.drawScreen( mouseX, mouseY, partialTicks ); |         super.render( mouseX, mouseY, partialTicks ); | ||||||
|         renderHoveredToolTip( mouseX, mouseY ); |         renderHoveredToolTip( mouseX, mouseY ); | ||||||
|     } |     } | ||||||
| } | } | ||||||
|   | |||||||
| @@ -8,29 +8,27 @@ package dan200.computercraft.client.gui; | |||||||
|  |  | ||||||
| import dan200.computercraft.ComputerCraft; | import dan200.computercraft.ComputerCraft; | ||||||
| import dan200.computercraft.client.gui.widgets.WidgetTerminal; | import dan200.computercraft.client.gui.widgets.WidgetTerminal; | ||||||
|  | import dan200.computercraft.client.gui.widgets.WidgetWrapper; | ||||||
| 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.turtle.blocks.TileTurtle; | import dan200.computercraft.shared.turtle.blocks.TileTurtle; | ||||||
| import dan200.computercraft.shared.turtle.inventory.ContainerTurtle; | import dan200.computercraft.shared.turtle.inventory.ContainerTurtle; | ||||||
| import net.minecraft.client.Minecraft; |  | ||||||
| import net.minecraft.client.gui.inventory.GuiContainer; | import net.minecraft.client.gui.inventory.GuiContainer; | ||||||
| import net.minecraft.client.renderer.GlStateManager; | import net.minecraft.client.renderer.GlStateManager; | ||||||
| import net.minecraft.util.ResourceLocation; | import net.minecraft.util.ResourceLocation; | ||||||
| import org.lwjgl.input.Keyboard; |  | ||||||
| import org.lwjgl.input.Mouse; |  | ||||||
|  |  | ||||||
| import java.io.IOException; |  | ||||||
|  |  | ||||||
| public class GuiTurtle extends GuiContainer | public class GuiTurtle extends GuiContainer | ||||||
| { | { | ||||||
|     private static final ResourceLocation BACKGROUND_NORMAL = new ResourceLocation( "computercraft", "textures/gui/turtle.png" ); |     private static final ResourceLocation BACKGROUND_NORMAL = new ResourceLocation( "computercraft", "textures/gui/turtle_normal.png" ); | ||||||
|     private static final ResourceLocation BACKGROUND_ADVANCED = new ResourceLocation( "computercraft", "textures/gui/turtle_advanced.png" ); |     private static final ResourceLocation BACKGROUND_ADVANCED = new ResourceLocation( "computercraft", "textures/gui/turtle_advanced.png" ); | ||||||
|  |  | ||||||
|     private ContainerTurtle m_container; |     private ContainerTurtle m_container; | ||||||
|  |  | ||||||
|     private final ComputerFamily m_family; |     private final ComputerFamily m_family; | ||||||
|     private final ClientComputer m_computer; |     private final ClientComputer m_computer; | ||||||
|     private WidgetTerminal m_terminalGui; |  | ||||||
|  |     private WidgetTerminal terminal; | ||||||
|  |     private WidgetWrapper terminalWrapper; | ||||||
|  |  | ||||||
|     public GuiTurtle( TileTurtle turtle, ContainerTurtle container ) |     public GuiTurtle( TileTurtle turtle, ContainerTurtle container ) | ||||||
|     { |     { | ||||||
| @@ -45,78 +43,48 @@ public class GuiTurtle extends GuiContainer | |||||||
|     } |     } | ||||||
|  |  | ||||||
|     @Override |     @Override | ||||||
|     public void initGui() |     protected void initGui() | ||||||
|     { |     { | ||||||
|         super.initGui(); |         super.initGui(); | ||||||
|         Keyboard.enableRepeatEvents( true ); |         mc.keyboardListener.enableRepeatEvents( true ); | ||||||
|         m_terminalGui = new WidgetTerminal( |  | ||||||
|             guiLeft + 8, |         int termPxWidth = ComputerCraft.terminalWidth_turtle * FixedWidthFontRenderer.FONT_WIDTH; | ||||||
|             guiTop + 8, |         int termPxHeight = ComputerCraft.terminalHeight_turtle * FixedWidthFontRenderer.FONT_HEIGHT; | ||||||
|  |  | ||||||
|  |         terminal = new WidgetTerminal( | ||||||
|  |             mc, () -> m_computer, | ||||||
|             ComputerCraft.terminalWidth_turtle, |             ComputerCraft.terminalWidth_turtle, | ||||||
|             ComputerCraft.terminalHeight_turtle, |             ComputerCraft.terminalHeight_turtle, | ||||||
|             () -> m_computer, |  | ||||||
|             2, 2, 2, 2 |             2, 2, 2, 2 | ||||||
|         ); |         ); | ||||||
|         m_terminalGui.setAllowFocusLoss( false ); |         terminalWrapper = new WidgetWrapper( terminal, 2 + 8 + guiLeft, 2 + 8 + guiTop, termPxWidth, termPxHeight ); | ||||||
|  |  | ||||||
|  |         children.add( terminalWrapper ); | ||||||
|  |         setFocused( terminalWrapper ); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @Override |     @Override | ||||||
|     public void onGuiClosed() |     public void onGuiClosed() | ||||||
|     { |     { | ||||||
|         super.onGuiClosed(); |         children.remove( terminal ); | ||||||
|         Keyboard.enableRepeatEvents( false ); |         terminal = null; | ||||||
|  |         mc.keyboardListener.enableRepeatEvents( false ); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @Override |     @Override | ||||||
|     public void updateScreen() |     public void tick() | ||||||
|     { |     { | ||||||
|         super.updateScreen(); |         super.tick(); | ||||||
|         m_terminalGui.update(); |         terminal.update(); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @Override |     private void drawSelectionSlot( boolean advanced ) | ||||||
|     protected void keyTyped( char c, int k ) throws IOException |  | ||||||
|     { |  | ||||||
|         if( k == 1 ) |  | ||||||
|         { |  | ||||||
|             super.keyTyped( c, k ); |  | ||||||
|         } |  | ||||||
|         else |  | ||||||
|         { |  | ||||||
|             if( m_terminalGui.onKeyTyped( c, k ) ) keyHandled = true; |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     @Override |  | ||||||
|     protected void mouseClicked( int x, int y, int button ) throws IOException |  | ||||||
|     { |  | ||||||
|         super.mouseClicked( x, y, button ); |  | ||||||
|         m_terminalGui.mouseClicked( x, y, button ); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     @Override |  | ||||||
|     public void handleMouseInput() throws IOException |  | ||||||
|     { |  | ||||||
|         super.handleMouseInput(); |  | ||||||
|         int x = Mouse.getEventX() * width / mc.displayWidth; |  | ||||||
|         int y = height - Mouse.getEventY() * height / mc.displayHeight - 1; |  | ||||||
|         m_terminalGui.handleMouseInput( x, y ); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     @Override |  | ||||||
|     public void handleKeyboardInput() throws IOException |  | ||||||
|     { |  | ||||||
|         super.handleKeyboardInput(); |  | ||||||
|         if( m_terminalGui.onKeyboardInput() ) keyHandled = true; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     protected void drawSelectionSlot( boolean advanced ) |  | ||||||
|     { |     { | ||||||
|         // Draw selection slot |         // Draw selection slot | ||||||
|         int slot = m_container.getSelectedSlot(); |         int slot = m_container.getSelectedSlot(); | ||||||
|         if( slot >= 0 ) |         if( slot >= 0 ) | ||||||
|         { |         { | ||||||
|             GlStateManager.color( 1.0F, 1.0F, 1.0F, 1.0F ); |             GlStateManager.color4f( 1.0F, 1.0F, 1.0F, 1.0F ); | ||||||
|             int slotX = slot % 4; |             int slotX = slot % 4; | ||||||
|             int slotY = slot / 4; |             int slotY = slot / 4; | ||||||
|             mc.getTextureManager().bindTexture( advanced ? BACKGROUND_ADVANCED : BACKGROUND_NORMAL ); |             mc.getTextureManager().bindTexture( advanced ? BACKGROUND_ADVANCED : BACKGROUND_NORMAL ); | ||||||
| @@ -129,10 +97,10 @@ public class GuiTurtle extends GuiContainer | |||||||
|     { |     { | ||||||
|         // Draw term |         // Draw term | ||||||
|         boolean advanced = m_family == ComputerFamily.Advanced; |         boolean advanced = m_family == ComputerFamily.Advanced; | ||||||
|         m_terminalGui.draw( Minecraft.getMinecraft(), 0, 0, mouseX, mouseY ); |         terminal.draw( terminalWrapper.getX(), terminalWrapper.getY() ); | ||||||
|  |  | ||||||
|         // Draw border/inventory |         // Draw border/inventory | ||||||
|         GlStateManager.color( 1.0F, 1.0F, 1.0F, 1.0F ); |         GlStateManager.color4f( 1.0F, 1.0F, 1.0F, 1.0F ); | ||||||
|         mc.getTextureManager().bindTexture( advanced ? BACKGROUND_ADVANCED : BACKGROUND_NORMAL ); |         mc.getTextureManager().bindTexture( advanced ? BACKGROUND_ADVANCED : BACKGROUND_NORMAL ); | ||||||
|         drawTexturedModalRect( guiLeft, guiTop, 0, 0, xSize, ySize ); |         drawTexturedModalRect( guiLeft, guiTop, 0, 0, xSize, ySize ); | ||||||
|  |  | ||||||
| @@ -140,10 +108,10 @@ public class GuiTurtle extends GuiContainer | |||||||
|     } |     } | ||||||
|  |  | ||||||
|     @Override |     @Override | ||||||
|     public void drawScreen( int mouseX, int mouseY, float partialTicks ) |     public void render( int mouseX, int mouseY, float partialTicks ) | ||||||
|     { |     { | ||||||
|         drawDefaultBackground(); |         drawDefaultBackground(); | ||||||
|         super.drawScreen( mouseX, mouseY, partialTicks ); |         super.render( mouseX, mouseY, partialTicks ); | ||||||
|         renderHoveredToolTip( mouseX, mouseY ); |         renderHoveredToolTip( mouseX, mouseY ); | ||||||
|     } |     } | ||||||
| } | } | ||||||
|   | |||||||
| @@ -1,84 +0,0 @@ | |||||||
| /* |  | ||||||
|  * This file is part of ComputerCraft - http://www.computercraft.info |  | ||||||
|  * Copyright Daniel Ratcliffe, 2011-2019. Do not distribute without permission. |  | ||||||
|  * Send enquiries to dratcliffe@gmail.com |  | ||||||
|  */ |  | ||||||
|  |  | ||||||
| package dan200.computercraft.client.gui.widgets; |  | ||||||
|  |  | ||||||
| import net.minecraft.client.Minecraft; |  | ||||||
| import net.minecraft.client.gui.Gui; |  | ||||||
|  |  | ||||||
| public abstract class Widget extends Gui |  | ||||||
| { |  | ||||||
|     private int m_xPosition; |  | ||||||
|     private int m_yPosition; |  | ||||||
|     private int m_width; |  | ||||||
|     private int m_height; |  | ||||||
|  |  | ||||||
|     protected Widget( int x, int y, int width, int height ) |  | ||||||
|     { |  | ||||||
|         m_xPosition = x; |  | ||||||
|         m_yPosition = y; |  | ||||||
|         m_width = width; |  | ||||||
|         m_height = height; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     public int getXPosition() |  | ||||||
|     { |  | ||||||
|         return m_xPosition; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     public int getYPosition() |  | ||||||
|     { |  | ||||||
|         return m_yPosition; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     public int getWidth() |  | ||||||
|     { |  | ||||||
|         return m_width; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     public int getHeight() |  | ||||||
|     { |  | ||||||
|         return m_height; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     public void update() |  | ||||||
|     { |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     public void draw( Minecraft mc, int xOrigin, int yOrigin, int mouseX, int mouseY ) |  | ||||||
|     { |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     public void handleMouseInput( int mouseX, int mouseY ) |  | ||||||
|     { |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     public boolean onKeyboardInput() |  | ||||||
|     { |  | ||||||
|         return false; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     @Deprecated |  | ||||||
|     public void handleKeyboardInput() |  | ||||||
|     { |  | ||||||
|         onKeyboardInput(); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     public void mouseClicked( int mouseX, int mouseY, int mouseButton ) |  | ||||||
|     { |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     public boolean onKeyTyped( char c, int k ) |  | ||||||
|     { |  | ||||||
|         return false; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     @Deprecated |  | ||||||
|     public void keyTyped( char c, int k ) |  | ||||||
|     { |  | ||||||
|         onKeyTyped( c, k ); |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| @@ -10,371 +10,349 @@ 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.core.terminal.TextBuffer; | ||||||
|  | import dan200.computercraft.shared.computer.core.ClientComputer; | ||||||
| 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.util.Colour; | 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.gui.GuiScreen; | import net.minecraft.client.gui.IGuiEventListener; | ||||||
|  | import net.minecraft.client.renderer.BufferBuilder; | ||||||
| import net.minecraft.client.renderer.GlStateManager; | import net.minecraft.client.renderer.GlStateManager; | ||||||
| import net.minecraft.util.ChatAllowedCharacters; | import net.minecraft.client.renderer.Tessellator; | ||||||
| import org.lwjgl.input.Keyboard; | import net.minecraft.client.renderer.vertex.DefaultVertexFormats; | ||||||
| import org.lwjgl.input.Mouse; | import net.minecraft.util.SharedConstants; | ||||||
|  | import org.lwjgl.glfw.GLFW; | ||||||
|  | import org.lwjgl.opengl.GL11; | ||||||
|  |  | ||||||
| import java.util.ArrayList; | import java.util.BitSet; | ||||||
|  | import java.util.function.Supplier; | ||||||
|  |  | ||||||
| import static dan200.computercraft.client.gui.FixedWidthFontRenderer.BACKGROUND; | import static dan200.computercraft.client.gui.FixedWidthFontRenderer.BACKGROUND; | ||||||
|  |  | ||||||
| public class WidgetTerminal extends Widget | public class WidgetTerminal implements IGuiEventListener | ||||||
| { | { | ||||||
|     private static final float TERMINATE_TIME = 0.5f; |     private static final float TERMINATE_TIME = 0.5f; | ||||||
|  |  | ||||||
|     private final IComputerContainer m_computer; |     private final Minecraft client; | ||||||
|  |  | ||||||
|     private float m_terminateTimer = 0.0f; |     private final Supplier<ClientComputer> computer; | ||||||
|     private float m_rebootTimer = 0.0f; |     private final int termWidth; | ||||||
|     private float m_shutdownTimer = 0.0f; |     private final int termHeight; | ||||||
|  |  | ||||||
|     private int m_lastClickButton = -1; |     private float terminateTimer = -1; | ||||||
|     private int m_lastClickX = -1; |     private float rebootTimer = -1; | ||||||
|     private int m_lastClickY = -1; |     private float shutdownTimer = -1; | ||||||
|  |  | ||||||
|     private boolean m_focus = false; |     private int lastMouseButton = -1; | ||||||
|     private boolean m_allowFocusLoss = true; |     private int lastMouseX = -1; | ||||||
|  |     private int lastMouseY = -1; | ||||||
|  |  | ||||||
|     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 BitSet keysDown = new BitSet( 256 ); | ||||||
|  |  | ||||||
|     public WidgetTerminal( int x, int y, int termWidth, int termHeight, IComputerContainer computer, int leftMargin, int rightMargin, int topMargin, int bottomMargin ) |     public WidgetTerminal( Minecraft client, Supplier<ClientComputer> computer, int termWidth, int termHeight, int leftMargin, int rightMargin, int topMargin, int bottomMargin ) | ||||||
|     { |     { | ||||||
|         super( |         this.client = client; | ||||||
|             x, y, |         this.computer = computer; | ||||||
|             leftMargin + rightMargin + termWidth * FixedWidthFontRenderer.FONT_WIDTH, |         this.termWidth = termWidth; | ||||||
|             topMargin + bottomMargin + termHeight * FixedWidthFontRenderer.FONT_HEIGHT |         this.termHeight = termHeight; | ||||||
|         ); |         this.leftMargin = leftMargin; | ||||||
|  |         this.rightMargin = rightMargin; | ||||||
|         m_computer = computer; |         this.topMargin = topMargin; | ||||||
|  |         this.bottomMargin = bottomMargin; | ||||||
|         m_leftMargin = leftMargin; |  | ||||||
|         m_rightMargin = rightMargin; |  | ||||||
|         m_topMargin = topMargin; |  | ||||||
|         m_bottomMargin = bottomMargin; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     public void setAllowFocusLoss( boolean allowFocusLoss ) |  | ||||||
|     { |  | ||||||
|         m_allowFocusLoss = allowFocusLoss; |  | ||||||
|         m_focus = m_focus || !allowFocusLoss; |  | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @Override |     @Override | ||||||
|     public boolean onKeyTyped( char ch, int key ) |     public boolean charTyped( char ch, int modifiers ) | ||||||
|     { |     { | ||||||
|         if( m_focus ) |         if( ch >= 32 && ch <= 126 || ch >= 160 && ch <= 255 ) // printable chars in byte range | ||||||
|         { |         { | ||||||
|             // Ctrl+V for paste |             // Queue the "char" event | ||||||
|             if( ch == 22 ) |             queueEvent( "char", Character.toString( ch ) ); | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         return true; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     @Override | ||||||
|  |     public boolean keyPressed( int key, int scancode, int modifiers ) | ||||||
|  |     { | ||||||
|  |         if( key == GLFW.GLFW_KEY_ESCAPE ) return false; | ||||||
|  |         if( (modifiers & GLFW.GLFW_MOD_CONTROL) != 0 ) | ||||||
|  |         { | ||||||
|  |             switch( key ) | ||||||
|             { |             { | ||||||
|                 String clipboard = GuiScreen.getClipboardString(); |                 case GLFW.GLFW_KEY_T: | ||||||
|                 if( clipboard != null ) |                     if( terminateTimer < 0 ) terminateTimer = 0; | ||||||
|                 { |                     return true; | ||||||
|                     // Clip to the first occurrence of \r or \n |                 case GLFW.GLFW_KEY_S: | ||||||
|                     int newLineIndex1 = clipboard.indexOf( '\r' ); |                     if( shutdownTimer < 0 ) shutdownTimer = 0; | ||||||
|                     int newLineIndex2 = clipboard.indexOf( '\n' ); |                     return true; | ||||||
|                     if( newLineIndex1 >= 0 && newLineIndex2 >= 0 ) |                 case GLFW.GLFW_KEY_R: | ||||||
|                     { |                     if( rebootTimer < 0 ) rebootTimer = 0; | ||||||
|                         clipboard = clipboard.substring( 0, Math.min( newLineIndex1, newLineIndex2 ) ); |                     return true; | ||||||
|                     } |  | ||||||
|                     else if( newLineIndex1 >= 0 ) |  | ||||||
|                     { |  | ||||||
|                         clipboard = clipboard.substring( 0, newLineIndex1 ); |  | ||||||
|                     } |  | ||||||
|                     else if( newLineIndex2 >= 0 ) |  | ||||||
|                     { |  | ||||||
|                         clipboard = clipboard.substring( 0, newLineIndex2 ); |  | ||||||
|                     } |  | ||||||
|  |  | ||||||
|                     // Filter the string |                 case GLFW.GLFW_KEY_V: | ||||||
|                     clipboard = ChatAllowedCharacters.filterAllowedCharacters( clipboard ); |                     // Ctrl+V for paste | ||||||
|  |                     String clipboard = client.keyboardListener.getClipboardString(); | ||||||
|                     if( !clipboard.isEmpty() ) |                     if( clipboard != null ) | ||||||
|                     { |                     { | ||||||
|                         // Clip to 512 characters |                         // Clip to the first occurrence of \r or \n | ||||||
|                         if( clipboard.length() > 512 ) |                         int newLineIndex1 = clipboard.indexOf( "\r" ); | ||||||
|  |                         int newLineIndex2 = clipboard.indexOf( "\n" ); | ||||||
|  |                         if( newLineIndex1 >= 0 && newLineIndex2 >= 0 ) | ||||||
|                         { |                         { | ||||||
|                             clipboard = clipboard.substring( 0, 512 ); |                             clipboard = clipboard.substring( 0, Math.min( newLineIndex1, newLineIndex2 ) ); | ||||||
|  |                         } | ||||||
|  |                         else if( newLineIndex1 >= 0 ) | ||||||
|  |                         { | ||||||
|  |                             clipboard = clipboard.substring( 0, newLineIndex1 ); | ||||||
|  |                         } | ||||||
|  |                         else if( newLineIndex2 >= 0 ) | ||||||
|  |                         { | ||||||
|  |                             clipboard = clipboard.substring( 0, newLineIndex2 ); | ||||||
|                         } |                         } | ||||||
|  |  | ||||||
|                         // Queue the "paste" event |                         // Filter the string | ||||||
|                         queueEvent( "paste", new Object[] { clipboard } ); |                         clipboard = SharedConstants.filterAllowedCharacters( clipboard ); | ||||||
|                     } |                         if( !clipboard.isEmpty() ) | ||||||
|                 } |                         { | ||||||
|                 return true; |                             // Clip to 512 characters and queue the event | ||||||
|             } |                             if( clipboard.length() > 512 ) clipboard = clipboard.substring( 0, 512 ); | ||||||
|  |                             queueEvent( "paste", clipboard ); | ||||||
|  |                         } | ||||||
|  |  | ||||||
|             // Regular keys normally |                         return true; | ||||||
|             if( m_terminateTimer <= 0.0f && m_rebootTimer <= 0.0f && m_shutdownTimer <= 0.0f ) |                     } | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         if( key >= 0 && terminateTimer < 0 && rebootTimer < 0 && shutdownTimer < 0 ) | ||||||
|  |         { | ||||||
|  |             // Queue the "key" event and add to the down set | ||||||
|  |             boolean repeat = keysDown.get( key ); | ||||||
|  |             keysDown.set( key ); | ||||||
|  |             IComputer computer = this.computer.get(); | ||||||
|  |             if( computer != null ) computer.keyDown( key, repeat ); | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         return true; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     @Override | ||||||
|  |     public boolean keyReleased( int key, int scancode, int modifiers ) | ||||||
|  |     { | ||||||
|  |         // Queue the "key_up" event and remove from the down set | ||||||
|  |         if( key >= 0 && keysDown.get( key ) ) | ||||||
|  |         { | ||||||
|  |             keysDown.set( key, false ); | ||||||
|  |             IComputer computer = this.computer.get(); | ||||||
|  |             if( computer != null ) computer.keyUp( key ); | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         switch( key ) | ||||||
|  |         { | ||||||
|  |             case GLFW.GLFW_KEY_T: | ||||||
|  |                 terminateTimer = -1; | ||||||
|  |                 break; | ||||||
|  |             case GLFW.GLFW_KEY_R: | ||||||
|  |                 rebootTimer = -1; | ||||||
|  |                 break; | ||||||
|  |             case GLFW.GLFW_KEY_S: | ||||||
|  |                 shutdownTimer = -1; | ||||||
|  |                 break; | ||||||
|  |             case GLFW.GLFW_KEY_LEFT_CONTROL: | ||||||
|  |             case GLFW.GLFW_KEY_RIGHT_CONTROL: | ||||||
|  |                 terminateTimer = rebootTimer = shutdownTimer = -1; | ||||||
|  |                 break; | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         return true; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     @Override | ||||||
|  |     public boolean mouseClicked( double mouseX, double mouseY, int button ) | ||||||
|  |     { | ||||||
|  |         ClientComputer computer = this.computer.get(); | ||||||
|  |         if( computer == null || !computer.isColour() || button < 0 || button > 2 ) return false; | ||||||
|  |  | ||||||
|  |         Terminal term = computer.getTerminal(); | ||||||
|  |         if( term != null ) | ||||||
|  |         { | ||||||
|  |             int charX = (int) (mouseX / FixedWidthFontRenderer.FONT_WIDTH); | ||||||
|  |             int charY = (int) (mouseY / FixedWidthFontRenderer.FONT_HEIGHT); | ||||||
|  |             charX = Math.min( Math.max( charX, 0 ), term.getWidth() - 1 ); | ||||||
|  |             charY = Math.min( Math.max( charY, 0 ), term.getHeight() - 1 ); | ||||||
|  |  | ||||||
|  |             computer.mouseClick( button + 1, charX + 1, charY + 1 ); | ||||||
|  |  | ||||||
|  |             lastMouseButton = button; | ||||||
|  |             lastMouseX = charX; | ||||||
|  |             lastMouseY = charY; | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         return true; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     @Override | ||||||
|  |     public boolean mouseReleased( double mouseX, double mouseY, int button ) | ||||||
|  |     { | ||||||
|  |         ClientComputer computer = this.computer.get(); | ||||||
|  |         if( computer == null || !computer.isColour() || button < 0 || button > 2 ) return false; | ||||||
|  |  | ||||||
|  |         Terminal term = computer.getTerminal(); | ||||||
|  |         if( term != null ) | ||||||
|  |         { | ||||||
|  |             int charX = (int) (mouseX / FixedWidthFontRenderer.FONT_WIDTH); | ||||||
|  |             int charY = (int) (mouseY / FixedWidthFontRenderer.FONT_HEIGHT); | ||||||
|  |             charX = Math.min( Math.max( charX, 0 ), term.getWidth() - 1 ); | ||||||
|  |             charY = Math.min( Math.max( charY, 0 ), term.getHeight() - 1 ); | ||||||
|  |  | ||||||
|  |             if( lastMouseButton == button ) | ||||||
|             { |             { | ||||||
|                 boolean repeat = Keyboard.isRepeatEvent(); |                 computer.mouseUp( lastMouseButton + 1, charX + 1, charY + 1 ); | ||||||
|                 boolean handled = false; |                 lastMouseButton = -1; | ||||||
|                 if( key > 0 ) |  | ||||||
|                 { |  | ||||||
|                     if( !repeat ) |  | ||||||
|                     { |  | ||||||
|                         m_keysDown.add( key ); |  | ||||||
|                     } |  | ||||||
|  |  | ||||||
|                     // Queue the "key" event |  | ||||||
|                     IComputer computer = m_computer.getComputer(); |  | ||||||
|                     if( computer != null ) computer.keyDown( key, repeat ); |  | ||||||
|                     handled = true; |  | ||||||
|                 } |  | ||||||
|  |  | ||||||
|                 if( (ch >= 32 && ch <= 126) || (ch >= 160 && ch <= 255) ) // printable chars in byte range |  | ||||||
|                 { |  | ||||||
|                     // Queue the "char" event |  | ||||||
|                     queueEvent( "char", new Object[] { Character.toString( ch ) } ); |  | ||||||
|                     handled = true; |  | ||||||
|                 } |  | ||||||
|  |  | ||||||
|                 return handled; |  | ||||||
|             } |             } | ||||||
|  |  | ||||||
|  |             lastMouseX = charX; | ||||||
|  |             lastMouseY = charY; | ||||||
|         } |         } | ||||||
|  |  | ||||||
|         return false; |         return false; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @Override |     @Override | ||||||
|     public void mouseClicked( int mouseX, int mouseY, int button ) |     public boolean mouseDragged( double mouseX, double mouseY, int button, double v2, double v3 ) | ||||||
|     { |     { | ||||||
|         if( mouseX >= getXPosition() && mouseX < getXPosition() + getWidth() && |         ClientComputer computer = this.computer.get(); | ||||||
|             mouseY >= getYPosition() && mouseY < getYPosition() + getHeight() ) |         if( computer == null || !computer.isColour() || button < 0 || button > 2 ) return false; | ||||||
|  |  | ||||||
|  |         Terminal term = computer.getTerminal(); | ||||||
|  |         if( term != null ) | ||||||
|         { |         { | ||||||
|             if( !m_focus && button == 0 ) |             int charX = (int) (mouseX / FixedWidthFontRenderer.FONT_WIDTH); | ||||||
|             { |             int charY = (int) (mouseY / FixedWidthFontRenderer.FONT_HEIGHT); | ||||||
|                 m_focus = true; |             charX = Math.min( Math.max( charX, 0 ), term.getWidth() - 1 ); | ||||||
|             } |             charY = Math.min( Math.max( charY, 0 ), term.getHeight() - 1 ); | ||||||
|  |  | ||||||
|             if( m_focus ) |             computer.mouseDrag( button + 1, charX + 1, charY + 1 ); | ||||||
|             { |  | ||||||
|                 IComputer computer = m_computer.getComputer(); |  | ||||||
|                 if( computer != null && computer.isColour() && button >= 0 && button <= 2 ) |  | ||||||
|                 { |  | ||||||
|                     Terminal term = computer.getTerminal(); |  | ||||||
|                     if( term != null ) |  | ||||||
|                     { |  | ||||||
|                         int charX = (mouseX - (getXPosition() + m_leftMargin)) / FixedWidthFontRenderer.FONT_WIDTH; |  | ||||||
|                         int charY = (mouseY - (getYPosition() + m_topMargin)) / FixedWidthFontRenderer.FONT_HEIGHT; |  | ||||||
|                         charX = Math.min( Math.max( charX, 0 ), term.getWidth() - 1 ); |  | ||||||
|                         charY = Math.min( Math.max( charY, 0 ), term.getHeight() - 1 ); |  | ||||||
|  |  | ||||||
|                         computer.mouseClick( button + 1, charX + 1, charY + 1 ); |             lastMouseX = charX; | ||||||
|  |             lastMouseY = charY; | ||||||
|                         m_lastClickButton = button; |             lastMouseButton = button; | ||||||
|                         m_lastClickX = charX; |  | ||||||
|                         m_lastClickY = charY; |  | ||||||
|                     } |  | ||||||
|                 } |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
|         else |  | ||||||
|         { |  | ||||||
|             if( m_focus && button == 0 && m_allowFocusLoss ) |  | ||||||
|             { |  | ||||||
|                 m_focus = false; |  | ||||||
|             } |  | ||||||
|         } |         } | ||||||
|  |  | ||||||
|  |         return false; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @Override |     @Override | ||||||
|     public boolean onKeyboardInput() |     public boolean mouseScrolled( double delta ) | ||||||
|     { |     { | ||||||
|         boolean handled = false; |         ClientComputer computer = this.computer.get(); | ||||||
|         for( int i = m_keysDown.size() - 1; i >= 0; --i ) |         if( computer == null || !computer.isColour() ) return false; | ||||||
|  |  | ||||||
|  |         if( lastMouseX >= 0 && lastMouseY >= 0 && delta != 0 ) | ||||||
|         { |         { | ||||||
|             int key = m_keysDown.get( i ); |             queueEvent( "mouse_scroll", delta < 0 ? 1 : -1, lastMouseX + 1, lastMouseY + 1 ); | ||||||
|             if( !Keyboard.isKeyDown( key ) ) |  | ||||||
|             { |  | ||||||
|                 m_keysDown.remove( i ); |  | ||||||
|                 if( m_focus ) |  | ||||||
|                 { |  | ||||||
|                     // Queue the "key_up" event |  | ||||||
|                     IComputer computer = m_computer.getComputer(); |  | ||||||
|                     if( computer != null ) computer.keyUp( key ); |  | ||||||
|                     handled = true; |  | ||||||
|                 } |  | ||||||
|             } |  | ||||||
|         } |         } | ||||||
|  |  | ||||||
|         return handled; |         return true; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @Override |  | ||||||
|     public void handleMouseInput( int mouseX, int mouseY ) |  | ||||||
|     { |  | ||||||
|         IComputer computer = m_computer.getComputer(); |  | ||||||
|         if( mouseX >= getXPosition() && mouseX < getXPosition() + getWidth() && |  | ||||||
|             mouseY >= getYPosition() && mouseY < getYPosition() + getHeight() && |  | ||||||
|             computer != null && computer.isColour() ) |  | ||||||
|         { |  | ||||||
|             Terminal term = computer.getTerminal(); |  | ||||||
|             if( term != null ) |  | ||||||
|             { |  | ||||||
|                 int charX = (mouseX - (getXPosition() + m_leftMargin)) / FixedWidthFontRenderer.FONT_WIDTH; |  | ||||||
|                 int charY = (mouseY - (getYPosition() + m_topMargin)) / FixedWidthFontRenderer.FONT_HEIGHT; |  | ||||||
|                 charX = Math.min( Math.max( charX, 0 ), term.getWidth() - 1 ); |  | ||||||
|                 charY = Math.min( Math.max( charY, 0 ), term.getHeight() - 1 ); |  | ||||||
|  |  | ||||||
|                 if( m_lastClickButton >= 0 && !Mouse.isButtonDown( m_lastClickButton ) ) |  | ||||||
|                 { |  | ||||||
|                     if( m_focus ) computer.mouseUp( m_lastClickButton + 1, charX + 1, charY + 1 ); |  | ||||||
|                     m_lastClickButton = -1; |  | ||||||
|                 } |  | ||||||
|  |  | ||||||
|                 int wheelChange = Mouse.getEventDWheel(); |  | ||||||
|                 if( wheelChange == 0 && m_lastClickButton == -1 ) |  | ||||||
|                 { |  | ||||||
|                     return; |  | ||||||
|                 } |  | ||||||
|  |  | ||||||
|                 if( m_focus ) |  | ||||||
|                 { |  | ||||||
|                     if( wheelChange < 0 ) |  | ||||||
|                     { |  | ||||||
|                         computer.mouseScroll( 1, charX + 1, charY + 1 ); |  | ||||||
|                     } |  | ||||||
|                     else if( wheelChange > 0 ) |  | ||||||
|                     { |  | ||||||
|                         computer.mouseScroll( -1, charX + 1, charY + 1 ); |  | ||||||
|                     } |  | ||||||
|  |  | ||||||
|                     if( m_lastClickButton >= 0 && (charX != m_lastClickX || charY != m_lastClickY) ) |  | ||||||
|                     { |  | ||||||
|                         computer.mouseDrag( m_lastClickButton + 1, charX + 1, charY + 1 ); |  | ||||||
|                         m_lastClickX = charX; |  | ||||||
|                         m_lastClickY = charY; |  | ||||||
|                     } |  | ||||||
|                 } |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     @Override |  | ||||||
|     public void update() |     public void update() | ||||||
|     { |     { | ||||||
|         // Handle special keys |         if( terminateTimer >= 0 && terminateTimer < TERMINATE_TIME && (terminateTimer += 0.05f) > TERMINATE_TIME ) | ||||||
|         if( m_focus && (Keyboard.isKeyDown( 29 ) || Keyboard.isKeyDown( 157 )) ) |  | ||||||
|         { |         { | ||||||
|             // Ctrl+T for terminate |             queueEvent( "terminate" ); | ||||||
|             if( Keyboard.isKeyDown( 20 ) ) |  | ||||||
|             { |  | ||||||
|                 if( m_terminateTimer < TERMINATE_TIME ) |  | ||||||
|                 { |  | ||||||
|                     m_terminateTimer += 0.05f; |  | ||||||
|                     if( m_terminateTimer >= TERMINATE_TIME ) queueEvent( "terminate" ); |  | ||||||
|                 } |  | ||||||
|             } |  | ||||||
|             else |  | ||||||
|             { |  | ||||||
|                 m_terminateTimer = 0.0f; |  | ||||||
|             } |  | ||||||
|  |  | ||||||
|             // Ctrl+R for reboot |  | ||||||
|             if( Keyboard.isKeyDown( 19 ) ) |  | ||||||
|             { |  | ||||||
|                 if( m_rebootTimer < TERMINATE_TIME ) |  | ||||||
|                 { |  | ||||||
|                     m_rebootTimer += 0.05f; |  | ||||||
|                     if( m_rebootTimer >= TERMINATE_TIME ) |  | ||||||
|                     { |  | ||||||
|                         IComputer computer = m_computer.getComputer(); |  | ||||||
|                         if( computer != null ) computer.reboot(); |  | ||||||
|                     } |  | ||||||
|                 } |  | ||||||
|             } |  | ||||||
|             else |  | ||||||
|             { |  | ||||||
|                 m_rebootTimer = 0.0f; |  | ||||||
|             } |  | ||||||
|  |  | ||||||
|             // Ctrl+S for shutdown |  | ||||||
|             if( Keyboard.isKeyDown( 31 ) ) |  | ||||||
|             { |  | ||||||
|                 if( m_shutdownTimer < TERMINATE_TIME ) |  | ||||||
|                 { |  | ||||||
|                     m_shutdownTimer += 0.05f; |  | ||||||
|                     if( m_shutdownTimer >= TERMINATE_TIME ) |  | ||||||
|                     { |  | ||||||
|                         IComputer computer = m_computer.getComputer(); |  | ||||||
|                         if( computer != null ) computer.shutdown(); |  | ||||||
|                     } |  | ||||||
|                 } |  | ||||||
|             } |  | ||||||
|             else |  | ||||||
|             { |  | ||||||
|                 m_shutdownTimer = 0.0f; |  | ||||||
|             } |  | ||||||
|         } |         } | ||||||
|         else |  | ||||||
|  |         if( shutdownTimer >= 0 && shutdownTimer < TERMINATE_TIME && (shutdownTimer += 0.05f) > TERMINATE_TIME ) | ||||||
|         { |         { | ||||||
|             m_terminateTimer = 0.0f; |             ClientComputer computer = this.computer.get(); | ||||||
|             m_rebootTimer = 0.0f; |             if( computer != null ) computer.shutdown(); | ||||||
|             m_shutdownTimer = 0.0f; |         } | ||||||
|  |  | ||||||
|  |         if( rebootTimer >= 0 && rebootTimer < TERMINATE_TIME && (rebootTimer += 0.05f) > TERMINATE_TIME ) | ||||||
|  |         { | ||||||
|  |             ClientComputer computer = this.computer.get(); | ||||||
|  |             if( computer != null ) computer.reboot(); | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @Override |     @Override | ||||||
|     public void draw( Minecraft mc, int xOrigin, int yOrigin, int mouseX, int mouseY ) |     public void focusChanged( boolean focused ) | ||||||
|     { |     { | ||||||
|         int startX = xOrigin + getXPosition(); |         if( !focused ) | ||||||
|         int startY = yOrigin + getYPosition(); |         { | ||||||
|  |             // When blurring, we should make all keys go up | ||||||
|  |             for( int key = 0; key < keysDown.size(); key++ ) | ||||||
|  |             { | ||||||
|  |                 if( keysDown.get( key ) ) queueEvent( "key_up", key ); | ||||||
|  |             } | ||||||
|  |             keysDown.clear(); | ||||||
|  |  | ||||||
|         synchronized( m_computer ) |             // When blurring, we should make the last mouse button go up | ||||||
|  |             if( lastMouseButton > 0 ) | ||||||
|  |             { | ||||||
|  |                 IComputer computer = this.computer.get(); | ||||||
|  |                 if( computer != null ) computer.mouseUp( lastMouseButton + 1, lastMouseX + 1, lastMouseY + 1 ); | ||||||
|  |                 lastMouseButton = -1; | ||||||
|  |             } | ||||||
|  |  | ||||||
|  |             shutdownTimer = terminateTimer = rebootTimer = -1; | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     public void draw( int originX, int originY ) | ||||||
|  |     { | ||||||
|  |         synchronized( computer ) | ||||||
|         { |         { | ||||||
|             // Draw the screen contents |             // Draw the screen contents | ||||||
|             IComputer computer = m_computer.getComputer(); |             ClientComputer computer = this.computer.get(); | ||||||
|             Terminal terminal = computer != null ? computer.getTerminal() : null; |             Terminal terminal = computer != null ? computer.getTerminal() : null; | ||||||
|             if( terminal != null ) |             if( terminal != null ) | ||||||
|             { |             { | ||||||
|                 // Draw the terminal |                 // Draw the terminal | ||||||
|                 boolean greyscale = !computer.isColour(); |                 boolean greyscale = !computer.isColour(); | ||||||
|  |  | ||||||
|                 Palette palette = terminal.getPalette(); |                 Palette palette = terminal.getPalette(); | ||||||
|  |  | ||||||
|                 // Get the data from the terminal first |                 // 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. |                 // Unfortunately we have to keep the lock for the whole of drawing, so the text doesn't change under us. | ||||||
|                 FixedWidthFontRenderer fontRenderer = FixedWidthFontRenderer.instance(); |                 FixedWidthFontRenderer fontRenderer = FixedWidthFontRenderer.instance(); | ||||||
|                 boolean tblink = m_focus && terminal.getCursorBlink() && FrameInfo.getGlobalCursorBlink(); |                 boolean tblink = terminal.getCursorBlink() && FrameInfo.getGlobalCursorBlink(); | ||||||
|                 int tw = terminal.getWidth(); |                 int tw = terminal.getWidth(); | ||||||
|                 int th = terminal.getHeight(); |                 int th = terminal.getHeight(); | ||||||
|                 int tx = terminal.getCursorX(); |                 int tx = terminal.getCursorX(); | ||||||
|                 int ty = terminal.getCursorY(); |                 int ty = terminal.getCursorY(); | ||||||
|  |  | ||||||
|                 int x = startX + m_leftMargin; |  | ||||||
|                 int y = startY + m_topMargin; |  | ||||||
|  |  | ||||||
|                 // Draw margins |                 // Draw margins | ||||||
|                 TextBuffer emptyLine = new TextBuffer( ' ', tw ); |                 TextBuffer emptyLine = new TextBuffer( ' ', tw ); | ||||||
|                 if( m_topMargin > 0 ) |                 if( topMargin > 0 ) | ||||||
|                 { |                 { | ||||||
|                     fontRenderer.drawString( emptyLine, x, startY, terminal.getTextColourLine( 0 ), terminal.getBackgroundColourLine( 0 ), m_leftMargin, m_rightMargin, greyscale, palette ); |                     fontRenderer.drawString( emptyLine, originX, originY - topMargin, | ||||||
|  |                         terminal.getTextColourLine( 0 ), terminal.getBackgroundColourLine( 0 ), | ||||||
|  |                         leftMargin, rightMargin, greyscale, palette ); | ||||||
|                 } |                 } | ||||||
|                 if( m_bottomMargin > 0 ) |  | ||||||
|  |                 if( 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 ); |                     fontRenderer.drawString( emptyLine, originX, originY + bottomMargin + (th - 1) * FixedWidthFontRenderer.FONT_HEIGHT, | ||||||
|  |                         terminal.getTextColourLine( th - 1 ), terminal.getBackgroundColourLine( th - 1 ), | ||||||
|  |                         leftMargin, rightMargin, greyscale, palette ); | ||||||
|                 } |                 } | ||||||
|  |  | ||||||
|                 // Draw lines |                 // Draw lines | ||||||
|  |                 int y = originY; | ||||||
|                 for( int line = 0; line < th; line++ ) |                 for( int line = 0; line < th; line++ ) | ||||||
|                 { |                 { | ||||||
|                     TextBuffer text = terminal.getLine( line ); |                     TextBuffer text = terminal.getLine( line ); | ||||||
|                     TextBuffer colour = terminal.getTextColourLine( line ); |                     TextBuffer colour = terminal.getTextColourLine( line ); | ||||||
|                     TextBuffer backgroundColour = terminal.getBackgroundColourLine( line ); |                     TextBuffer backgroundColour = terminal.getBackgroundColourLine( line ); | ||||||
|                     fontRenderer.drawString( text, x, y, colour, backgroundColour, m_leftMargin, m_rightMargin, greyscale, palette ); |                     fontRenderer.drawString( text, originX, y, colour, backgroundColour, leftMargin, rightMargin, greyscale, palette ); | ||||||
|                     y += FixedWidthFontRenderer.FONT_HEIGHT; |                     y += FixedWidthFontRenderer.FONT_HEIGHT; | ||||||
|                 } |                 } | ||||||
|  |  | ||||||
| @@ -385,8 +363,8 @@ public class WidgetTerminal extends Widget | |||||||
|  |  | ||||||
|                     fontRenderer.drawString( |                     fontRenderer.drawString( | ||||||
|                         cursor, |                         cursor, | ||||||
|                         x + FixedWidthFontRenderer.FONT_WIDTH * tx, |                         originX + FixedWidthFontRenderer.FONT_WIDTH * tx, | ||||||
|                         startY + m_topMargin + FixedWidthFontRenderer.FONT_HEIGHT * ty, |                         originY + FixedWidthFontRenderer.FONT_HEIGHT * ty, | ||||||
|                         cursorColour, null, |                         cursorColour, null, | ||||||
|                         0, 0, |                         0, 0, | ||||||
|                         greyscale, |                         greyscale, | ||||||
| @@ -397,16 +375,29 @@ public class WidgetTerminal extends Widget | |||||||
|             else |             else | ||||||
|             { |             { | ||||||
|                 // Draw a black background |                 // Draw a black background | ||||||
|                 mc.getTextureManager().bindTexture( BACKGROUND ); |  | ||||||
|                 Colour black = Colour.Black; |                 Colour black = Colour.Black; | ||||||
|                 GlStateManager.color( black.getR(), black.getG(), black.getB(), 1.0f ); |                 GlStateManager.color4f( black.getR(), black.getG(), black.getB(), 1.0f ); | ||||||
|                 try |                 try | ||||||
|                 { |                 { | ||||||
|                     drawTexturedModalRect( startX, startY, 0, 0, getWidth(), getHeight() ); |                     int x = originX - leftMargin; | ||||||
|  |                     int y = originY - rightMargin; | ||||||
|  |                     int width = termWidth * FixedWidthFontRenderer.FONT_WIDTH + leftMargin + rightMargin; | ||||||
|  |                     int height = termHeight * FixedWidthFontRenderer.FONT_HEIGHT + topMargin + bottomMargin; | ||||||
|  |  | ||||||
|  |                     client.getTextureManager().bindTexture( BACKGROUND ); | ||||||
|  |  | ||||||
|  |                     Tessellator tesslector = Tessellator.getInstance(); | ||||||
|  |                     BufferBuilder buffer = tesslector.getBuffer(); | ||||||
|  |                     buffer.begin( GL11.GL_QUADS, DefaultVertexFormats.POSITION_TEX ); | ||||||
|  |                     buffer.pos( x, y + height, 0 ).tex( 0 / 256.0, height / 256.0 ).endVertex(); | ||||||
|  |                     buffer.pos( x + width, y + height, 0 ).tex( width / 256.0, height / 256.0 ).endVertex(); | ||||||
|  |                     buffer.pos( x + width, y, 0 ).tex( width / 256.0, 0 / 256.0 ).endVertex(); | ||||||
|  |                     buffer.pos( x, y, 0 ).tex( 0 / 256.0, 0 / 256.0 ).endVertex(); | ||||||
|  |                     tesslector.draw(); | ||||||
|                 } |                 } | ||||||
|                 finally |                 finally | ||||||
|                 { |                 { | ||||||
|                     GlStateManager.color( 1.0f, 1.0f, 1.0f, 1.0f ); |                     GlStateManager.color4f( 1.0f, 1.0f, 1.0f, 1.0f ); | ||||||
|                 } |                 } | ||||||
|             } |             } | ||||||
|         } |         } | ||||||
| @@ -414,13 +405,13 @@ public class WidgetTerminal extends Widget | |||||||
|  |  | ||||||
|     private void queueEvent( String event ) |     private void queueEvent( String event ) | ||||||
|     { |     { | ||||||
|         IComputer computer = m_computer.getComputer(); |         ClientComputer computer = this.computer.get(); | ||||||
|         if( computer != null ) computer.queueEvent( event ); |         if( computer != null ) computer.queueEvent( event ); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     private void queueEvent( String event, Object[] args ) |     private void queueEvent( String event, Object... args ) | ||||||
|     { |     { | ||||||
|         IComputer computer = m_computer.getComputer(); |         ClientComputer computer = this.computer.get(); | ||||||
|         if( computer != null ) computer.queueEvent( event, args ); |         if( computer != null ) computer.queueEvent( event, args ); | ||||||
|     } |     } | ||||||
| } | } | ||||||
|   | |||||||
| @@ -0,0 +1,104 @@ | |||||||
|  | /* | ||||||
|  |  * This file is part of ComputerCraft - http://www.computercraft.info | ||||||
|  |  * Copyright Daniel Ratcliffe, 2011-2019. Do not distribute without permission. | ||||||
|  |  * Send enquiries to dratcliffe@gmail.com | ||||||
|  |  */ | ||||||
|  |  | ||||||
|  | package dan200.computercraft.client.gui.widgets; | ||||||
|  |  | ||||||
|  | import net.minecraft.client.gui.IGuiEventListener; | ||||||
|  |  | ||||||
|  | public class WidgetWrapper implements IGuiEventListener | ||||||
|  | { | ||||||
|  |     private final IGuiEventListener listener; | ||||||
|  |     private final int x; | ||||||
|  |     private final int y; | ||||||
|  |     private final int width; | ||||||
|  |     private final int height; | ||||||
|  |  | ||||||
|  |     public WidgetWrapper( IGuiEventListener listener, int x, int y, int width, int height ) | ||||||
|  |     { | ||||||
|  |         this.listener = listener; | ||||||
|  |         this.x = x; | ||||||
|  |         this.y = y; | ||||||
|  |         this.width = width; | ||||||
|  |         this.height = height; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     @Override | ||||||
|  |     public void focusChanged( boolean b ) | ||||||
|  |     { | ||||||
|  |         listener.focusChanged( b ); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     @Override | ||||||
|  |     public boolean canFocus() | ||||||
|  |     { | ||||||
|  |         return listener.canFocus(); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     @Override | ||||||
|  |     public boolean mouseClicked( double x, double y, int button ) | ||||||
|  |     { | ||||||
|  |         double dx = x - this.x, dy = y - this.y; | ||||||
|  |         return dx >= 0 && dx < width && dy >= 0 && dy < height && listener.mouseClicked( dx, dy, button ); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     @Override | ||||||
|  |     public boolean mouseReleased( double x, double y, int button ) | ||||||
|  |     { | ||||||
|  |         double dx = x - this.x, dy = y - this.y; | ||||||
|  |         return dx >= 0 && dx < width && dy >= 0 && dy < height && listener.mouseReleased( dx, dy, button ); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     @Override | ||||||
|  |     public boolean mouseDragged( double x, double y, int button, double deltaX, double deltaY ) | ||||||
|  |     { | ||||||
|  |         double dx = x - this.x, dy = y - this.y; | ||||||
|  |         return dx >= 0 && dx < width && dy >= 0 && dy < height && listener.mouseDragged( dx, dy, button, deltaX, deltaY ); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     @Override | ||||||
|  |     public boolean mouseScrolled( double delta ) | ||||||
|  |     { | ||||||
|  |         return listener.mouseScrolled( delta ); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     @Override | ||||||
|  |     public boolean keyPressed( int key, int scancode, int modifiers ) | ||||||
|  |     { | ||||||
|  |         return listener.keyPressed( key, scancode, modifiers ); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     @Override | ||||||
|  |     public boolean keyReleased( int key, int scancode, int modifiers ) | ||||||
|  |     { | ||||||
|  |         return listener.keyReleased( key, scancode, modifiers ); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     @Override | ||||||
|  |     public boolean charTyped( char character, int modifiers ) | ||||||
|  |     { | ||||||
|  |         return listener.charTyped( character, modifiers ); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     public int getX() | ||||||
|  |     { | ||||||
|  |         return x; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     public int getY() | ||||||
|  |     { | ||||||
|  |         return y; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     public int getWidth() | ||||||
|  |     { | ||||||
|  |         return width; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     public int getHeight() | ||||||
|  |     { | ||||||
|  |         return height; | ||||||
|  |     } | ||||||
|  | } | ||||||
| @@ -7,56 +7,89 @@ | |||||||
| package dan200.computercraft.client.proxy; | package dan200.computercraft.client.proxy; | ||||||
|  |  | ||||||
| import dan200.computercraft.ComputerCraft; | import dan200.computercraft.ComputerCraft; | ||||||
|  | import dan200.computercraft.client.gui.*; | ||||||
| import dan200.computercraft.client.render.TileEntityCableRenderer; | import dan200.computercraft.client.render.TileEntityCableRenderer; | ||||||
| import dan200.computercraft.client.render.TileEntityMonitorRenderer; | import dan200.computercraft.client.render.TileEntityMonitorRenderer; | ||||||
| import dan200.computercraft.client.render.TileEntityTurtleRenderer; | import dan200.computercraft.client.render.TileEntityTurtleRenderer; | ||||||
| import dan200.computercraft.shared.command.CommandCopy; | import dan200.computercraft.shared.computer.blocks.TileComputer; | ||||||
|  | import dan200.computercraft.shared.computer.core.ClientComputer; | ||||||
|  | import dan200.computercraft.shared.computer.inventory.ContainerViewComputer; | ||||||
|  | import dan200.computercraft.shared.network.container.*; | ||||||
| import dan200.computercraft.shared.peripheral.modem.wired.TileCable; | import dan200.computercraft.shared.peripheral.modem.wired.TileCable; | ||||||
| import dan200.computercraft.shared.peripheral.monitor.ClientMonitor; | import dan200.computercraft.shared.peripheral.monitor.ClientMonitor; | ||||||
| import dan200.computercraft.shared.peripheral.monitor.TileMonitor; | import dan200.computercraft.shared.peripheral.monitor.TileMonitor; | ||||||
| import dan200.computercraft.shared.proxy.ComputerCraftProxyCommon; |  | ||||||
| import dan200.computercraft.shared.turtle.blocks.TileTurtle; | import dan200.computercraft.shared.turtle.blocks.TileTurtle; | ||||||
| import net.minecraftforge.client.ClientCommandHandler; | import dan200.computercraft.shared.turtle.inventory.ContainerTurtle; | ||||||
|  | import net.minecraft.client.Minecraft; | ||||||
|  | import net.minecraft.client.gui.inventory.GuiContainer; | ||||||
|  | import net.minecraft.entity.player.EntityPlayer; | ||||||
|  | import net.minecraftforge.api.distmarker.Dist; | ||||||
| import net.minecraftforge.event.world.WorldEvent; | import net.minecraftforge.event.world.WorldEvent; | ||||||
|  | import net.minecraftforge.eventbus.api.SubscribeEvent; | ||||||
|  | import net.minecraftforge.fml.ExtensionPoint; | ||||||
|  | import net.minecraftforge.fml.ModLoadingContext; | ||||||
| import net.minecraftforge.fml.client.registry.ClientRegistry; | import net.minecraftforge.fml.client.registry.ClientRegistry; | ||||||
| import net.minecraftforge.fml.common.Mod; | import net.minecraftforge.fml.common.Mod; | ||||||
| import net.minecraftforge.fml.common.eventhandler.SubscribeEvent; | import net.minecraftforge.fml.event.lifecycle.FMLClientSetupEvent; | ||||||
| import net.minecraftforge.fml.relauncher.Side; |  | ||||||
|  |  | ||||||
| public class ComputerCraftProxyClient extends ComputerCraftProxyCommon | import java.util.function.BiFunction; | ||||||
|  |  | ||||||
|  | @Mod.EventBusSubscriber( modid = ComputerCraft.MOD_ID, value = Dist.CLIENT, bus = Mod.EventBusSubscriber.Bus.MOD ) | ||||||
|  | public final class ComputerCraftProxyClient | ||||||
| { | { | ||||||
|     @Override |     @SubscribeEvent | ||||||
|     public void preInit() |     public static void setupClient( FMLClientSetupEvent event ) | ||||||
|     { |     { | ||||||
|         super.preInit(); |         registerContainers(); | ||||||
|  |  | ||||||
|         // Register any client-specific commands |         // Setup TESRs | ||||||
|         ClientCommandHandler.instance.registerCommand( CommandCopy.INSTANCE ); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     @Override |  | ||||||
|     public void init() |  | ||||||
|     { |  | ||||||
|         super.init(); |  | ||||||
|  |  | ||||||
|         // Setup renderers |  | ||||||
|         ClientRegistry.bindTileEntitySpecialRenderer( TileMonitor.class, new TileEntityMonitorRenderer() ); |         ClientRegistry.bindTileEntitySpecialRenderer( TileMonitor.class, new TileEntityMonitorRenderer() ); | ||||||
|         ClientRegistry.bindTileEntitySpecialRenderer( TileCable.class, new TileEntityCableRenderer() ); |         ClientRegistry.bindTileEntitySpecialRenderer( TileCable.class, new TileEntityCableRenderer() ); | ||||||
|         ClientRegistry.bindTileEntitySpecialRenderer( TileTurtle.class, new TileEntityTurtleRenderer() ); |         ClientRegistry.bindTileEntitySpecialRenderer( TileTurtle.class, new TileEntityTurtleRenderer() ); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @Mod.EventBusSubscriber( modid = ComputerCraft.MOD_ID, value = Side.CLIENT ) |     private static void registerContainers() | ||||||
|  |     { | ||||||
|  |         ContainerType.registerGui( TileEntityContainerType::computer, ( packet, player ) -> | ||||||
|  |             new GuiComputer( (TileComputer) packet.getTileEntity( player ) ) ); | ||||||
|  |         ContainerType.registerGui( TileEntityContainerType::diskDrive, GuiDiskDrive::new ); | ||||||
|  |         ContainerType.registerGui( TileEntityContainerType::printer, GuiPrinter::new ); | ||||||
|  |         ContainerType.registerGui( TileEntityContainerType::turtle, ( packet, player ) -> { | ||||||
|  |             TileTurtle turtle = (TileTurtle) packet.getTileEntity( player ); | ||||||
|  |             return new GuiTurtle( turtle, new ContainerTurtle( player.inventory, turtle.getAccess(), turtle.getClientComputer() ) ); | ||||||
|  |         } ); | ||||||
|  |  | ||||||
|  |         ContainerType.registerGui( PocketComputerContainerType::new, GuiPocketComputer::new ); | ||||||
|  |         ContainerType.registerGui( PrintoutContainerType::new, GuiPrintout::new ); | ||||||
|  |         ContainerType.registerGui( ViewComputerContainerType::new, ( packet, player ) -> { | ||||||
|  |             ClientComputer computer = ComputerCraft.clientComputerRegistry.get( packet.instanceId ); | ||||||
|  |             if( computer == null ) | ||||||
|  |             { | ||||||
|  |                 ComputerCraft.clientComputerRegistry.add( packet.instanceId, computer = new ClientComputer( packet.instanceId ) ); | ||||||
|  |             } | ||||||
|  |  | ||||||
|  |             ContainerViewComputer container = new ContainerViewComputer( computer ); | ||||||
|  |             return new GuiComputer( container, packet.family, computer, packet.width, packet.height ); | ||||||
|  |         } ); | ||||||
|  |  | ||||||
|  |         ModLoadingContext.get().registerExtensionPoint( ExtensionPoint.GUIFACTORY, () -> packet -> { | ||||||
|  |             ContainerType<?> type = ContainerType.factories.get( packet.getId() ).get(); | ||||||
|  |             if( packet.getAdditionalData() != null ) type.fromBytes( packet.getAdditionalData() ); | ||||||
|  |             return ((BiFunction<ContainerType<?>, EntityPlayer, GuiContainer>) ContainerType.guiFactories.get( packet.getId() )) | ||||||
|  |                 .apply( type, Minecraft.getInstance().player ); | ||||||
|  |         } ); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     @Mod.EventBusSubscriber( modid = ComputerCraft.MOD_ID, value = Dist.CLIENT ) | ||||||
|     public static final class ForgeHandlers |     public static final class ForgeHandlers | ||||||
|     { |     { | ||||||
|         @SubscribeEvent |         @SubscribeEvent | ||||||
|         public static void onWorldUnload( WorldEvent.Unload event ) |         public static void onWorldUnload( WorldEvent.Unload event ) | ||||||
|         { |         { | ||||||
|             if( event.getWorld().isRemote ) |             if( event.getWorld().isRemote() ) | ||||||
|             { |             { | ||||||
|                 ClientMonitor.destroyAll(); |                 ClientMonitor.destroyAll(); | ||||||
|             } |             } | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |  | ||||||
| } | } | ||||||
|   | |||||||
| @@ -7,8 +7,8 @@ | |||||||
| package dan200.computercraft.client.render; | package dan200.computercraft.client.render; | ||||||
|  |  | ||||||
| import net.minecraft.client.Minecraft; | import net.minecraft.client.Minecraft; | ||||||
|  | import net.minecraft.client.renderer.FirstPersonRenderer; | ||||||
| import net.minecraft.client.renderer.GlStateManager; | import net.minecraft.client.renderer.GlStateManager; | ||||||
| import net.minecraft.client.renderer.ItemRenderer; |  | ||||||
| import net.minecraft.entity.player.EntityPlayer; | import net.minecraft.entity.player.EntityPlayer; | ||||||
| import net.minecraft.item.ItemStack; | import net.minecraft.item.ItemStack; | ||||||
| import net.minecraft.util.EnumHand; | import net.minecraft.util.EnumHand; | ||||||
| @@ -21,13 +21,13 @@ public abstract class ItemMapLikeRenderer | |||||||
|      * The main rendering method for the item |      * The main rendering method for the item | ||||||
|      * |      * | ||||||
|      * @param stack The stack to render |      * @param stack The stack to render | ||||||
|      * @see ItemRenderer#renderMapFirstPerson(ItemStack) |      * @see FirstPersonRenderer#renderMapFirstPerson(ItemStack) | ||||||
|      */ |      */ | ||||||
|     protected abstract void renderItem( ItemStack stack ); |     protected abstract void renderItem( ItemStack stack ); | ||||||
|  |  | ||||||
|     protected void renderItemFirstPerson( EnumHand hand, float pitch, float equipProgress, float swingProgress, ItemStack stack ) |     protected void renderItemFirstPerson( EnumHand hand, float pitch, float equipProgress, float swingProgress, ItemStack stack ) | ||||||
|     { |     { | ||||||
|         EntityPlayer player = Minecraft.getMinecraft().player; |         EntityPlayer player = Minecraft.getInstance().player; | ||||||
|  |  | ||||||
|         GlStateManager.pushMatrix(); |         GlStateManager.pushMatrix(); | ||||||
|         if( hand == EnumHand.MAIN_HAND && player.getHeldItemOffhand().isEmpty() ) |         if( hand == EnumHand.MAIN_HAND && player.getHeldItemOffhand().isEmpty() ) | ||||||
| @@ -51,35 +51,35 @@ public abstract class ItemMapLikeRenderer | |||||||
|      * @param equipProgress The equip progress of this item |      * @param equipProgress The equip progress of this item | ||||||
|      * @param swingProgress The swing progress of this item |      * @param swingProgress The swing progress of this item | ||||||
|      * @param stack         The stack to render |      * @param stack         The stack to render | ||||||
|      * @see ItemRenderer#renderMapFirstPersonSide(float, EnumHandSide, float, ItemStack) |      * @see FirstPersonRenderer#renderMapFirstPersonSide(float, EnumHandSide, float, ItemStack) | ||||||
|      */ |      */ | ||||||
|     private void renderItemFirstPersonSide( EnumHandSide side, float equipProgress, float swingProgress, ItemStack stack ) |     private void renderItemFirstPersonSide( EnumHandSide side, float equipProgress, float swingProgress, ItemStack stack ) | ||||||
|     { |     { | ||||||
|         Minecraft minecraft = Minecraft.getMinecraft(); |         Minecraft minecraft = Minecraft.getInstance(); | ||||||
|         float offset = side == EnumHandSide.RIGHT ? 1f : -1f; |         float offset = side == EnumHandSide.RIGHT ? 1f : -1f; | ||||||
|         GlStateManager.translate( offset * 0.125f, -0.125f, 0f ); |         GlStateManager.translatef( offset * 0.125f, -0.125f, 0f ); | ||||||
|  |  | ||||||
|         // If the player is not invisible then render a single arm |         // If the player is not invisible then render a single arm | ||||||
|         if( !minecraft.player.isInvisible() ) |         if( !minecraft.player.isInvisible() ) | ||||||
|         { |         { | ||||||
|             GlStateManager.pushMatrix(); |             GlStateManager.pushMatrix(); | ||||||
|             GlStateManager.rotate( offset * 10f, 0f, 0f, 1f ); |             GlStateManager.rotatef( offset * 10f, 0f, 0f, 1f ); | ||||||
|             minecraft.getItemRenderer().renderArmFirstPerson( equipProgress, swingProgress, side ); |             minecraft.getFirstPersonRenderer().renderArmFirstPerson( equipProgress, swingProgress, side ); | ||||||
|             GlStateManager.popMatrix(); |             GlStateManager.popMatrix(); | ||||||
|         } |         } | ||||||
|  |  | ||||||
|         // Setup the appropriate transformations. This is just copied from the |         // Setup the appropriate transformations. This is just copied from the | ||||||
|         // corresponding method in ItemRenderer. |         // corresponding method in ItemRenderer. | ||||||
|         GlStateManager.pushMatrix(); |         GlStateManager.pushMatrix(); | ||||||
|         GlStateManager.translate( offset * 0.51f, -0.08f + equipProgress * -1.2f, -0.75f ); |         GlStateManager.translatef( offset * 0.51f, -0.08f + equipProgress * -1.2f, -0.75f ); | ||||||
|         float f1 = MathHelper.sqrt( swingProgress ); |         float f1 = MathHelper.sqrt( swingProgress ); | ||||||
|         float f2 = MathHelper.sin( f1 * (float) Math.PI ); |         float f2 = MathHelper.sin( f1 * (float) Math.PI ); | ||||||
|         float f3 = -0.5f * f2; |         float f3 = -0.5f * f2; | ||||||
|         float f4 = 0.4f * MathHelper.sin( f1 * ((float) Math.PI * 2f) ); |         float f4 = 0.4f * MathHelper.sin( f1 * ((float) Math.PI * 2f) ); | ||||||
|         float f5 = -0.3f * MathHelper.sin( swingProgress * (float) Math.PI ); |         float f5 = -0.3f * MathHelper.sin( swingProgress * (float) Math.PI ); | ||||||
|         GlStateManager.translate( offset * f3, f4 - 0.3f * f2, f5 ); |         GlStateManager.translatef( offset * f3, f4 - 0.3f * f2, f5 ); | ||||||
|         GlStateManager.rotate( f2 * -45f, 1f, 0f, 0f ); |         GlStateManager.rotatef( f2 * -45f, 1f, 0f, 0f ); | ||||||
|         GlStateManager.rotate( offset * f2 * -30f, 0f, 1f, 0f ); |         GlStateManager.rotatef( offset * f2 * -30f, 0f, 1f, 0f ); | ||||||
|  |  | ||||||
|         renderItem( stack ); |         renderItem( stack ); | ||||||
|  |  | ||||||
| @@ -93,25 +93,25 @@ public abstract class ItemMapLikeRenderer | |||||||
|      * @param equipProgress The equip progress of this item |      * @param equipProgress The equip progress of this item | ||||||
|      * @param swingProgress The swing progress of this item |      * @param swingProgress The swing progress of this item | ||||||
|      * @param stack         The stack to render |      * @param stack         The stack to render | ||||||
|      * @see ItemRenderer#renderMapFirstPerson(float, float, float) |      * @see FirstPersonRenderer#renderMapFirstPerson(float, float, float) | ||||||
|      */ |      */ | ||||||
|     private void renderItemFirstPersonCenter( float pitch, float equipProgress, float swingProgress, ItemStack stack ) |     private void renderItemFirstPersonCenter( float pitch, float equipProgress, float swingProgress, ItemStack stack ) | ||||||
|     { |     { | ||||||
|         ItemRenderer itemRenderer = Minecraft.getMinecraft().getItemRenderer(); |         FirstPersonRenderer renderer = Minecraft.getInstance().getFirstPersonRenderer(); | ||||||
|  |  | ||||||
|         // Setup the appropriate transformations. This is just copied from the |         // Setup the appropriate transformations. This is just copied from the | ||||||
|         // corresponding method in ItemRenderer. |         // corresponding method in ItemRenderer. | ||||||
|         float swingRt = MathHelper.sqrt( swingProgress ); |         float swingRt = MathHelper.sqrt( swingProgress ); | ||||||
|         float tX = -0.2f * MathHelper.sin( swingProgress * (float) Math.PI ); |         float tX = -0.2f * MathHelper.sin( swingProgress * (float) Math.PI ); | ||||||
|         float tZ = -0.4f * MathHelper.sin( swingRt * (float) Math.PI ); |         float tZ = -0.4f * MathHelper.sin( swingRt * (float) Math.PI ); | ||||||
|         GlStateManager.translate( 0f, -tX / 2f, tZ ); |         GlStateManager.translatef( 0f, -tX / 2f, tZ ); | ||||||
|         float pitchAngle = itemRenderer.getMapAngleFromPitch( pitch ); |         float pitchAngle = renderer.getMapAngleFromPitch( pitch ); | ||||||
|         GlStateManager.translate( 0f, 0.04f + equipProgress * -1.2f + pitchAngle * -0.5f, -0.72f ); |         GlStateManager.translatef( 0f, 0.04f + equipProgress * -1.2f + pitchAngle * -0.5f, -0.72f ); | ||||||
|         GlStateManager.rotate( pitchAngle * -85f, 1f, 0f, 0f ); |         GlStateManager.rotatef( pitchAngle * -85f, 1f, 0f, 0f ); | ||||||
|         itemRenderer.renderArms(); |         renderer.renderArms(); | ||||||
|         float rX = MathHelper.sin( swingRt * (float) Math.PI ); |         float rX = MathHelper.sin( swingRt * (float) Math.PI ); | ||||||
|         GlStateManager.rotate( rX * 20f, 1f, 0f, 0f ); |         GlStateManager.rotatef( rX * 20f, 1f, 0f, 0f ); | ||||||
|         GlStateManager.scale( 2f, 2f, 2f ); |         GlStateManager.scalef( 2f, 2f, 2f ); | ||||||
|  |  | ||||||
|         renderItem( stack ); |         renderItem( stack ); | ||||||
|     } |     } | ||||||
|   | |||||||
| @@ -16,17 +16,17 @@ import dan200.computercraft.shared.pocket.items.ItemPocketComputer; | |||||||
| 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.GlStateManager; | import net.minecraft.client.renderer.GlStateManager; | ||||||
| import net.minecraft.client.renderer.RenderItem; | import net.minecraft.client.renderer.ItemRenderer; | ||||||
| import net.minecraft.client.renderer.block.model.IBakedModel; | import net.minecraft.client.renderer.model.IBakedModel; | ||||||
| import net.minecraft.client.renderer.block.model.ItemCameraTransforms; | import net.minecraft.client.renderer.model.ItemCameraTransforms.TransformType; | ||||||
| import net.minecraft.client.renderer.texture.TextureManager; | import net.minecraft.client.renderer.texture.TextureManager; | ||||||
| import net.minecraft.client.renderer.texture.TextureMap; | import net.minecraft.client.renderer.texture.TextureMap; | ||||||
| import net.minecraft.item.ItemStack; | import net.minecraft.item.ItemStack; | ||||||
|  | import net.minecraftforge.api.distmarker.Dist; | ||||||
| import net.minecraftforge.client.ForgeHooksClient; | import net.minecraftforge.client.ForgeHooksClient; | ||||||
| import net.minecraftforge.client.event.RenderSpecificHandEvent; | import net.minecraftforge.client.event.RenderSpecificHandEvent; | ||||||
|  | import net.minecraftforge.eventbus.api.SubscribeEvent; | ||||||
| import net.minecraftforge.fml.common.Mod; | import net.minecraftforge.fml.common.Mod; | ||||||
| import net.minecraftforge.fml.common.eventhandler.SubscribeEvent; |  | ||||||
| import net.minecraftforge.fml.relauncher.Side; |  | ||||||
| import org.lwjgl.opengl.GL11; | import org.lwjgl.opengl.GL11; | ||||||
|  |  | ||||||
| import static dan200.computercraft.client.gui.FixedWidthFontRenderer.FONT_HEIGHT; | import static dan200.computercraft.client.gui.FixedWidthFontRenderer.FONT_HEIGHT; | ||||||
| @@ -35,7 +35,7 @@ import static dan200.computercraft.client.gui.FixedWidthFontRenderer.FONT_WIDTH; | |||||||
| /** | /** | ||||||
|  * Emulates map rendering for pocket computers |  * Emulates map rendering for pocket computers | ||||||
|  */ |  */ | ||||||
| @Mod.EventBusSubscriber( modid = ComputerCraft.MOD_ID, value = Side.CLIENT ) | @Mod.EventBusSubscriber( modid = ComputerCraft.MOD_ID, value = Dist.CLIENT ) | ||||||
| public final class ItemPocketRenderer extends ItemMapLikeRenderer | public final class ItemPocketRenderer extends ItemMapLikeRenderer | ||||||
| { | { | ||||||
|     private static final ItemPocketRenderer INSTANCE = new ItemPocketRenderer(); |     private static final ItemPocketRenderer INSTANCE = new ItemPocketRenderer(); | ||||||
| @@ -61,9 +61,9 @@ public final class ItemPocketRenderer extends ItemMapLikeRenderer | |||||||
|         // in ItemRenderer |         // in ItemRenderer | ||||||
|         GlStateManager.disableLighting(); |         GlStateManager.disableLighting(); | ||||||
|  |  | ||||||
|         GlStateManager.rotate( 180f, 0f, 1f, 0f ); |         GlStateManager.rotatef( 180f, 0f, 1f, 0f ); | ||||||
|         GlStateManager.rotate( 180f, 0f, 0f, 1f ); |         GlStateManager.rotatef( 180f, 0f, 0f, 1f ); | ||||||
|         GlStateManager.scale( 0.5, 0.5, 0.5 ); |         GlStateManager.scalef( 0.5f, 0.5f, 0.5f ); | ||||||
|  |  | ||||||
|         ClientComputer computer = ItemPocketComputer.createClientComputer( stack ); |         ClientComputer computer = ItemPocketComputer.createClientComputer( stack ); | ||||||
|  |  | ||||||
| @@ -72,28 +72,28 @@ public final class ItemPocketRenderer extends ItemMapLikeRenderer | |||||||
|             // we display the pocket light and other such decorations. |             // we display the pocket light and other such decorations. | ||||||
|             GlStateManager.pushMatrix(); |             GlStateManager.pushMatrix(); | ||||||
|  |  | ||||||
|             GlStateManager.scale( 1.0f, -1.0f, 1.0f ); |             GlStateManager.scalef( 1.0f, -1.0f, 1.0f ); | ||||||
|  |  | ||||||
|             Minecraft minecraft = Minecraft.getMinecraft(); |             Minecraft minecraft = Minecraft.getInstance(); | ||||||
|             TextureManager textureManager = minecraft.getTextureManager(); |             TextureManager textureManager = minecraft.getTextureManager(); | ||||||
|             RenderItem renderItem = minecraft.getRenderItem(); |             ItemRenderer renderItem = minecraft.getItemRenderer(); | ||||||
|  |  | ||||||
|             // Copy of RenderItem#renderItemModelIntoGUI but without the translation or scaling |             // Copy of RenderItem#renderItemModelIntoGUI but without the translation or scaling | ||||||
|             textureManager.bindTexture( TextureMap.LOCATION_BLOCKS_TEXTURE ); |             textureManager.bindTexture( TextureMap.LOCATION_BLOCKS_TEXTURE ); | ||||||
|             textureManager.getTexture( TextureMap.LOCATION_BLOCKS_TEXTURE ).setBlurMipmap( false, false ); |             textureManager.getTexture( TextureMap.LOCATION_BLOCKS_TEXTURE ).setBlurMipmap( false, false ); | ||||||
|  |  | ||||||
|             GlStateManager.enableRescaleNormal(); |             GlStateManager.enableRescaleNormal(); | ||||||
|             GlStateManager.enableAlpha(); |             GlStateManager.enableAlphaTest(); | ||||||
|             GlStateManager.alphaFunc( GL11.GL_GREATER, 0.1F ); |             GlStateManager.alphaFunc( GL11.GL_GREATER, 0.1F ); | ||||||
|             GlStateManager.enableBlend(); |             GlStateManager.enableBlend(); | ||||||
|             GlStateManager.blendFunc( GlStateManager.SourceFactor.SRC_ALPHA, GlStateManager.DestFactor.ONE_MINUS_SRC_ALPHA ); |             GlStateManager.blendFunc( GlStateManager.SourceFactor.SRC_ALPHA, GlStateManager.DestFactor.ONE_MINUS_SRC_ALPHA ); | ||||||
|             GlStateManager.color( 1.0F, 1.0F, 1.0F, 1.0F ); |             GlStateManager.color4f( 1.0F, 1.0F, 1.0F, 1.0F ); | ||||||
|  |  | ||||||
|             IBakedModel baked = renderItem.getItemModelWithOverrides( stack, null, null ); |             IBakedModel baked = renderItem.getItemModelWithOverrides( stack, null, null ); | ||||||
|             baked = ForgeHooksClient.handleCameraTransforms( baked, ItemCameraTransforms.TransformType.GUI, false ); |             baked = ForgeHooksClient.handleCameraTransforms( baked, TransformType.GUI, false ); | ||||||
|             renderItem.renderItem( stack, baked ); |             renderItem.renderItem( stack, baked ); | ||||||
|  |  | ||||||
|             GlStateManager.disableAlpha(); |             GlStateManager.disableAlphaTest(); | ||||||
|             GlStateManager.disableRescaleNormal(); |             GlStateManager.disableRescaleNormal(); | ||||||
|  |  | ||||||
|             GlStateManager.popMatrix(); |             GlStateManager.popMatrix(); | ||||||
| @@ -108,13 +108,13 @@ public final class ItemPocketRenderer extends ItemMapLikeRenderer | |||||||
|                 synchronized( terminal ) |                 synchronized( terminal ) | ||||||
|                 { |                 { | ||||||
|                     GlStateManager.pushMatrix(); |                     GlStateManager.pushMatrix(); | ||||||
|                     GlStateManager.disableDepth(); |                     GlStateManager.disableDepthTest(); | ||||||
|  |  | ||||||
|                     // Reset the position to be at the top left corner of the pocket computer |                     // Reset the position to be at the top left corner of the pocket computer | ||||||
|                     // Note we translate towards the screen slightly too. |                     // Note we translate towards the screen slightly too. | ||||||
|                     GlStateManager.translate( -8 / 16.0, -8 / 16.0, 0.5 / 16.0 ); |                     GlStateManager.translated( -8 / 16.0, -8 / 16.0, 0.5 / 16.0 ); | ||||||
|                     // Translate to the top left of the screen. |                     // Translate to the top left of the screen. | ||||||
|                     GlStateManager.translate( 4 / 16.0, 3 / 16.0, 0 ); |                     GlStateManager.translated( 4 / 16.0, 3 / 16.0, 0 ); | ||||||
|  |  | ||||||
|                     // Work out the scaling required to resize the terminal in order to fit on the computer |                     // Work out the scaling required to resize the terminal in order to fit on the computer | ||||||
|                     final int margin = 2; |                     final int margin = 2; | ||||||
| @@ -126,7 +126,7 @@ public final class ItemPocketRenderer extends ItemMapLikeRenderer | |||||||
|  |  | ||||||
|                     // The grid is 8 * 8 wide, so we start with a base of 1/2 (8 / 16). |                     // The grid is 8 * 8 wide, so we start with a base of 1/2 (8 / 16). | ||||||
|                     double scale = 1.0 / 2.0 / max; |                     double scale = 1.0 / 2.0 / max; | ||||||
|                     GlStateManager.scale( scale, scale, scale ); |                     GlStateManager.scaled( scale, scale, scale ); | ||||||
|  |  | ||||||
|                     // The margin/start positions are determined in order for the terminal to be centred. |                     // The margin/start positions are determined in order for the terminal to be centred. | ||||||
|                     int startX = (max - width) / 2 + margin; |                     int startX = (max - width) / 2 + margin; | ||||||
| @@ -160,7 +160,7 @@ public final class ItemPocketRenderer extends ItemMapLikeRenderer | |||||||
|                         ); |                         ); | ||||||
|                     } |                     } | ||||||
|  |  | ||||||
|                     GlStateManager.enableDepth(); |                     GlStateManager.enableDepthTest(); | ||||||
|                     GlStateManager.popMatrix(); |                     GlStateManager.popMatrix(); | ||||||
|                 } |                 } | ||||||
|             } |             } | ||||||
|   | |||||||
| @@ -10,11 +10,11 @@ import dan200.computercraft.ComputerCraft; | |||||||
| import dan200.computercraft.shared.media.items.ItemPrintout; | import dan200.computercraft.shared.media.items.ItemPrintout; | ||||||
| import net.minecraft.client.renderer.GlStateManager; | import net.minecraft.client.renderer.GlStateManager; | ||||||
| import net.minecraft.item.ItemStack; | import net.minecraft.item.ItemStack; | ||||||
|  | import net.minecraftforge.api.distmarker.Dist; | ||||||
| import net.minecraftforge.client.event.RenderItemInFrameEvent; | import net.minecraftforge.client.event.RenderItemInFrameEvent; | ||||||
| import net.minecraftforge.client.event.RenderSpecificHandEvent; | import net.minecraftforge.client.event.RenderSpecificHandEvent; | ||||||
|  | import net.minecraftforge.eventbus.api.SubscribeEvent; | ||||||
| import net.minecraftforge.fml.common.Mod; | import net.minecraftforge.fml.common.Mod; | ||||||
| import net.minecraftforge.fml.common.eventhandler.SubscribeEvent; |  | ||||||
| import net.minecraftforge.fml.relauncher.Side; |  | ||||||
|  |  | ||||||
| import static dan200.computercraft.client.gui.FixedWidthFontRenderer.FONT_HEIGHT; | import static dan200.computercraft.client.gui.FixedWidthFontRenderer.FONT_HEIGHT; | ||||||
| import static dan200.computercraft.client.gui.FixedWidthFontRenderer.FONT_WIDTH; | import static dan200.computercraft.client.gui.FixedWidthFontRenderer.FONT_WIDTH; | ||||||
| @@ -25,7 +25,7 @@ import static dan200.computercraft.shared.media.items.ItemPrintout.LINE_MAX_LENG | |||||||
| /** | /** | ||||||
|  * Emulates map and item-frame rendering for printouts |  * Emulates map and item-frame rendering for printouts | ||||||
|  */ |  */ | ||||||
| @Mod.EventBusSubscriber( modid = ComputerCraft.MOD_ID, value = Side.CLIENT ) | @Mod.EventBusSubscriber( modid = ComputerCraft.MOD_ID, value = Dist.CLIENT ) | ||||||
| public final class ItemPrintoutRenderer extends ItemMapLikeRenderer | public final class ItemPrintoutRenderer extends ItemMapLikeRenderer | ||||||
| { | { | ||||||
|     private static final ItemPrintoutRenderer INSTANCE = new ItemPrintoutRenderer(); |     private static final ItemPrintoutRenderer INSTANCE = new ItemPrintoutRenderer(); | ||||||
| @@ -38,10 +38,9 @@ public final class ItemPrintoutRenderer extends ItemMapLikeRenderer | |||||||
|     public static void onRenderInHand( RenderSpecificHandEvent event ) |     public static void onRenderInHand( RenderSpecificHandEvent event ) | ||||||
|     { |     { | ||||||
|         ItemStack stack = event.getItemStack(); |         ItemStack stack = event.getItemStack(); | ||||||
|         if( stack.getItem() != ComputerCraft.Items.printout ) return; |         if( !(stack.getItem() instanceof ItemPrintout) ) return; | ||||||
|  |  | ||||||
|         event.setCanceled( true ); |         event.setCanceled( true ); | ||||||
|  |  | ||||||
|         INSTANCE.renderItemFirstPerson( event.getHand(), event.getInterpolatedPitch(), event.getEquipProgress(), event.getSwingProgress(), event.getItemStack() ); |         INSTANCE.renderItemFirstPerson( event.getHand(), event.getInterpolatedPitch(), event.getEquipProgress(), event.getSwingProgress(), event.getItemStack() ); | ||||||
|     } |     } | ||||||
|  |  | ||||||
| @@ -49,13 +48,13 @@ public final class ItemPrintoutRenderer extends ItemMapLikeRenderer | |||||||
|     protected void renderItem( ItemStack stack ) |     protected void renderItem( ItemStack stack ) | ||||||
|     { |     { | ||||||
|         // Setup various transformations. Note that these are partially adapated from the corresponding method |         // Setup various transformations. Note that these are partially adapated from the corresponding method | ||||||
|         // in ItemRenderer.renderMapFirstPerson |         // in FirstPersonRenderer.renderFirstPersonMap | ||||||
|         GlStateManager.disableLighting(); |         GlStateManager.disableLighting(); | ||||||
|  |  | ||||||
|         GlStateManager.rotate( 180f, 0f, 1f, 0f ); |         GlStateManager.rotatef( 180f, 0f, 1f, 0f ); | ||||||
|         GlStateManager.rotate( 180f, 0f, 0f, 1f ); |         GlStateManager.rotatef( 180f, 0f, 0f, 1f ); | ||||||
|         GlStateManager.scale( 0.42f, 0.42f, -0.42f ); |         GlStateManager.scalef( 0.42f, 0.42f, -0.42f ); | ||||||
|         GlStateManager.translate( -0.5f, -0.48f, 0.0f ); |         GlStateManager.translatef( -0.5f, -0.48f, 0.0f ); | ||||||
|  |  | ||||||
|         drawPrintout( stack ); |         drawPrintout( stack ); | ||||||
|  |  | ||||||
| @@ -66,17 +65,17 @@ public final class ItemPrintoutRenderer extends ItemMapLikeRenderer | |||||||
|     public static void onRenderInFrame( RenderItemInFrameEvent event ) |     public static void onRenderInFrame( RenderItemInFrameEvent event ) | ||||||
|     { |     { | ||||||
|         ItemStack stack = event.getItem(); |         ItemStack stack = event.getItem(); | ||||||
|         if( stack.getItem() != ComputerCraft.Items.printout ) return; |         if( !(stack.getItem() instanceof ItemPrintout) ) return; | ||||||
|  |  | ||||||
|         event.setCanceled( true ); |         event.setCanceled( true ); | ||||||
|  |  | ||||||
|         GlStateManager.disableLighting(); |         GlStateManager.disableLighting(); | ||||||
|  |  | ||||||
|         // Move a little bit forward to ensure we're not clipping with the frame |         // Move a little bit forward to ensure we're not clipping with the frame | ||||||
|         GlStateManager.translate( 0.0f, 0.0f, -0.001f ); |         GlStateManager.translatef( 0.0f, 0.0f, -0.001f ); | ||||||
|         GlStateManager.rotate( 180f, 0f, 0f, 1f ); |         GlStateManager.rotatef( 180f, 0f, 0f, 1f ); | ||||||
|         GlStateManager.scale( 0.95f, 0.95f, -0.95f ); |         GlStateManager.scalef( 0.95f, 0.95f, -0.95f ); | ||||||
|         GlStateManager.translate( -0.5f, -0.5f, 0.0f ); |         GlStateManager.translatef( -0.5f, -0.5f, 0.0f ); | ||||||
|  |  | ||||||
|         drawPrintout( stack ); |         drawPrintout( stack ); | ||||||
|  |  | ||||||
| @@ -87,7 +86,7 @@ public final class ItemPrintoutRenderer extends ItemMapLikeRenderer | |||||||
|     private static void drawPrintout( ItemStack stack ) |     private static void drawPrintout( ItemStack stack ) | ||||||
|     { |     { | ||||||
|         int pages = ItemPrintout.getPageCount( stack ); |         int pages = ItemPrintout.getPageCount( stack ); | ||||||
|         boolean book = ItemPrintout.getType( stack ) == ItemPrintout.Type.Book; |         boolean book = ((ItemPrintout) stack.getItem()).getType() == ItemPrintout.Type.BOOK; | ||||||
|  |  | ||||||
|         double width = LINE_MAX_LENGTH * FONT_WIDTH + X_TEXT_MARGIN * 2; |         double width = LINE_MAX_LENGTH * FONT_WIDTH + X_TEXT_MARGIN * 2; | ||||||
|         double height = LINES_PER_PAGE * FONT_HEIGHT + Y_TEXT_MARGIN * 2; |         double height = LINES_PER_PAGE * FONT_HEIGHT + Y_TEXT_MARGIN * 2; | ||||||
| @@ -108,8 +107,8 @@ public final class ItemPrintoutRenderer extends ItemMapLikeRenderer | |||||||
|  |  | ||||||
|         // Scale the printout to fit correctly. |         // Scale the printout to fit correctly. | ||||||
|         double scale = 1.0 / max; |         double scale = 1.0 / max; | ||||||
|         GlStateManager.scale( scale, scale, scale ); |         GlStateManager.scaled( scale, scale, scale ); | ||||||
|         GlStateManager.translate( (max - width) / 2.0f, (max - height) / 2.0f, 0.0f ); |         GlStateManager.translated( (max - width) / 2.0, (max - height) / 2.0, 0.0 ); | ||||||
|  |  | ||||||
|         drawBorder( 0, 0, -0.01, 0, pages, book ); |         drawBorder( 0, 0, -0.01, 0, pages, book ); | ||||||
|         drawText( X_TEXT_MARGIN, Y_TEXT_MARGIN, 0, ItemPrintout.getText( stack ), ItemPrintout.getColours( stack ) ); |         drawText( X_TEXT_MARGIN, Y_TEXT_MARGIN, 0, ItemPrintout.getText( stack ), ItemPrintout.getColours( stack ) ); | ||||||
|   | |||||||
| @@ -6,7 +6,7 @@ | |||||||
|  |  | ||||||
| package dan200.computercraft.client.render; | package dan200.computercraft.client.render; | ||||||
|  |  | ||||||
| import net.minecraft.client.renderer.block.model.BakedQuad; | import net.minecraft.client.renderer.model.BakedQuad; | ||||||
| import net.minecraft.client.renderer.texture.TextureAtlasSprite; | import net.minecraft.client.renderer.texture.TextureAtlasSprite; | ||||||
| import net.minecraft.client.renderer.vertex.VertexFormat; | import net.minecraft.client.renderer.vertex.VertexFormat; | ||||||
| import net.minecraft.util.EnumFacing; | import net.minecraft.util.EnumFacing; | ||||||
|   | |||||||
| @@ -74,10 +74,10 @@ public final class PrintoutRenderer | |||||||
|  |  | ||||||
|     public static void drawText( int x, int y, int start, String[] text, String[] colours ) |     public static void drawText( int x, int y, int start, String[] text, String[] colours ) | ||||||
|     { |     { | ||||||
|         GlStateManager.color( 1.0f, 1.0f, 1.0f, 1.0f ); |         GlStateManager.color4f( 1.0f, 1.0f, 1.0f, 1.0f ); | ||||||
|         GlStateManager.enableBlend(); |         GlStateManager.enableBlend(); | ||||||
|         GlStateManager.enableTexture2D(); |         GlStateManager.enableTexture2D(); | ||||||
|         GlStateManager.tryBlendFuncSeparate( SourceFactor.SRC_ALPHA, DestFactor.ONE_MINUS_SRC_ALPHA, SourceFactor.ONE, DestFactor.ZERO ); |         GlStateManager.blendFuncSeparate( SourceFactor.SRC_ALPHA, DestFactor.ONE_MINUS_SRC_ALPHA, SourceFactor.ONE, DestFactor.ZERO ); | ||||||
|  |  | ||||||
|         FixedWidthFontRenderer fontRenderer = FixedWidthFontRenderer.instance(); |         FixedWidthFontRenderer fontRenderer = FixedWidthFontRenderer.instance(); | ||||||
|  |  | ||||||
| @@ -89,12 +89,12 @@ public final class PrintoutRenderer | |||||||
|  |  | ||||||
|     public static void drawBorder( double x, double y, double z, int page, int pages, boolean isBook ) |     public static void drawBorder( double x, double y, double z, int page, int pages, boolean isBook ) | ||||||
|     { |     { | ||||||
|         GlStateManager.color( 1.0f, 1.0f, 1.0f, 1.0f ); |         GlStateManager.color4f( 1.0f, 1.0f, 1.0f, 1.0f ); | ||||||
|         GlStateManager.enableBlend(); |         GlStateManager.enableBlend(); | ||||||
|         GlStateManager.enableTexture2D(); |         GlStateManager.enableTexture2D(); | ||||||
|         GlStateManager.tryBlendFuncSeparate( SourceFactor.SRC_ALPHA, DestFactor.ONE_MINUS_SRC_ALPHA, SourceFactor.ONE, DestFactor.ZERO ); |         GlStateManager.blendFuncSeparate( SourceFactor.SRC_ALPHA, DestFactor.ONE_MINUS_SRC_ALPHA, SourceFactor.ONE, DestFactor.ZERO ); | ||||||
|  |  | ||||||
|         Minecraft.getMinecraft().getTextureManager().bindTexture( BG ); |         Minecraft.getInstance().getTextureManager().bindTexture( BG ); | ||||||
|  |  | ||||||
|         Tessellator tessellator = Tessellator.getInstance(); |         Tessellator tessellator = Tessellator.getInstance(); | ||||||
|         BufferBuilder buffer = tessellator.getBuffer(); |         BufferBuilder buffer = tessellator.getBuffer(); | ||||||
|   | |||||||
| @@ -7,216 +7,81 @@ | |||||||
| package dan200.computercraft.client.render; | package dan200.computercraft.client.render; | ||||||
|  |  | ||||||
| import dan200.computercraft.ComputerCraft; | import dan200.computercraft.ComputerCraft; | ||||||
| import dan200.computercraft.shared.peripheral.PeripheralType; |  | ||||||
| import dan200.computercraft.shared.peripheral.modem.wired.BlockCable; | import dan200.computercraft.shared.peripheral.modem.wired.BlockCable; | ||||||
| import dan200.computercraft.shared.peripheral.modem.wired.CableBounds; | import dan200.computercraft.shared.peripheral.modem.wired.CableShapes; | ||||||
| import dan200.computercraft.shared.util.WorldUtil; | import dan200.computercraft.shared.util.WorldUtil; | ||||||
| import net.minecraft.block.state.IBlockState; | import net.minecraft.block.state.IBlockState; | ||||||
| import net.minecraft.client.renderer.BufferBuilder; | import net.minecraft.client.Minecraft; | ||||||
| import net.minecraft.client.renderer.GlStateManager; | import net.minecraft.client.renderer.GlStateManager; | ||||||
| import net.minecraft.client.renderer.RenderGlobal; | import net.minecraft.client.renderer.WorldRenderer; | ||||||
| import net.minecraft.client.renderer.Tessellator; |  | ||||||
| import net.minecraft.client.renderer.vertex.DefaultVertexFormats; |  | ||||||
| import net.minecraft.entity.player.EntityPlayer; | import net.minecraft.entity.player.EntityPlayer; | ||||||
| import net.minecraft.util.EnumFacing; |  | ||||||
| import net.minecraft.util.math.BlockPos; | import net.minecraft.util.math.BlockPos; | ||||||
| import net.minecraft.util.math.RayTraceResult; | import net.minecraft.util.math.RayTraceResult; | ||||||
|  | import net.minecraft.util.math.shapes.VoxelShape; | ||||||
| import net.minecraft.world.World; | import net.minecraft.world.World; | ||||||
|  | import net.minecraftforge.api.distmarker.Dist; | ||||||
| import net.minecraftforge.client.event.DrawBlockHighlightEvent; | import net.minecraftforge.client.event.DrawBlockHighlightEvent; | ||||||
|  | import net.minecraftforge.eventbus.api.SubscribeEvent; | ||||||
| import net.minecraftforge.fml.common.Mod; | import net.minecraftforge.fml.common.Mod; | ||||||
| import net.minecraftforge.fml.common.eventhandler.SubscribeEvent; |  | ||||||
| import net.minecraftforge.fml.relauncher.Side; |  | ||||||
| import org.lwjgl.opengl.GL11; | import org.lwjgl.opengl.GL11; | ||||||
|  |  | ||||||
| @Mod.EventBusSubscriber( modid = ComputerCraft.MOD_ID, value = Side.CLIENT ) | @Mod.EventBusSubscriber( modid = ComputerCraft.MOD_ID, value = Dist.CLIENT ) | ||||||
| public final class RenderOverlayCable | public final class RenderOverlayCable | ||||||
| { | { | ||||||
|     private static final float EXPAND = 0.002f; |  | ||||||
|     private static final double MIN = CableBounds.MIN - EXPAND; |  | ||||||
|     private static final double MAX = CableBounds.MAX + EXPAND; |  | ||||||
|  |  | ||||||
|     private RenderOverlayCable() |     private RenderOverlayCable() | ||||||
|     { |     { | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * Draw an outline for a specific part of a cable "Multipart". | ||||||
|  |      * | ||||||
|  |      * @param event The event to observe | ||||||
|  |      * @see WorldRenderer#drawSelectionBox(EntityPlayer, RayTraceResult, int, float) | ||||||
|  |      */ | ||||||
|     @SubscribeEvent |     @SubscribeEvent | ||||||
|     public static void drawHighlight( DrawBlockHighlightEvent event ) |     public static void drawHighlight( DrawBlockHighlightEvent event ) | ||||||
|     { |     { | ||||||
|         if( event.getTarget().typeOfHit != RayTraceResult.Type.BLOCK ) return; |         if( event.getTarget().type != RayTraceResult.Type.BLOCK ) return; | ||||||
|  |  | ||||||
|         BlockPos pos = event.getTarget().getBlockPos(); |         BlockPos pos = event.getTarget().getBlockPos(); | ||||||
|         World world = event.getPlayer().getEntityWorld(); |         World world = event.getPlayer().getEntityWorld(); | ||||||
|  |  | ||||||
|         IBlockState state = world.getBlockState( pos ); |         IBlockState state = world.getBlockState( pos ); | ||||||
|         if( state.getBlock() != ComputerCraft.Blocks.cable ) return; |  | ||||||
|  |  | ||||||
|         state = state.getActualState( world, pos ); |         // We only care about instances with both cable and modem. | ||||||
|  |         if( state.getBlock() != ComputerCraft.Blocks.cable || state.get( BlockCable.MODEM ).getFacing() == null || !state.get( BlockCable.CABLE ) ) | ||||||
|  |         { | ||||||
|  |             return; | ||||||
|  |         } | ||||||
|  |  | ||||||
|         event.setCanceled( true ); |         event.setCanceled( true ); | ||||||
|         PeripheralType type = BlockCable.getPeripheralType( state ); |  | ||||||
|  |         EntityPlayer player = event.getPlayer(); | ||||||
|  |         Minecraft mc = Minecraft.getInstance(); | ||||||
|  |         float partialTicks = event.getPartialTicks(); | ||||||
|  |  | ||||||
|         GlStateManager.enableBlend(); |         GlStateManager.enableBlend(); | ||||||
|         GlStateManager.tryBlendFuncSeparate( GL11.GL_SRC_ALPHA, GL11.GL_ONE_MINUS_SRC_ALPHA, 1, 0 ); |         GlStateManager.blendFuncSeparate( GlStateManager.SourceFactor.SRC_ALPHA, GlStateManager.DestFactor.ONE_MINUS_SRC_ALPHA, GlStateManager.SourceFactor.ONE, GlStateManager.DestFactor.ZERO ); | ||||||
|         GlStateManager.color( 0.0f, 0.0f, 0.0f, 0.4f ); |         GlStateManager.lineWidth( Math.max( 2.5F, mc.mainWindow.getFramebufferWidth() / 1920.0F * 2.5F ) ); | ||||||
|         GL11.glLineWidth( 2.0F ); |  | ||||||
|         GlStateManager.disableTexture2D(); |         GlStateManager.disableTexture2D(); | ||||||
|         GlStateManager.depthMask( false ); |         GlStateManager.depthMask( false ); | ||||||
|  |         GlStateManager.matrixMode( GL11.GL_PROJECTION ); | ||||||
|         GlStateManager.pushMatrix(); |         GlStateManager.pushMatrix(); | ||||||
|  |         GlStateManager.scalef( 1.0F, 1.0F, 0.999F ); | ||||||
|  |  | ||||||
|         { |         double x = player.lastTickPosX + (player.posX - player.lastTickPosX) * partialTicks; | ||||||
|             EntityPlayer player = event.getPlayer(); |         double y = player.lastTickPosY + (player.posY - player.lastTickPosY) * partialTicks; | ||||||
|             double x = player.lastTickPosX + (player.posX - player.lastTickPosX) * event.getPartialTicks(); |         double z = player.lastTickPosZ + (player.posZ - player.lastTickPosZ) * partialTicks; | ||||||
|             double y = player.lastTickPosY + (player.posY - player.lastTickPosY) * event.getPartialTicks(); |  | ||||||
|             double z = player.lastTickPosZ + (player.posZ - player.lastTickPosZ) * event.getPartialTicks(); |  | ||||||
|  |  | ||||||
|             GlStateManager.translate( -x + pos.getX(), -y + pos.getY(), -z + pos.getZ() ); |         VoxelShape shape = WorldUtil.isVecInside( CableShapes.getModemShape( state ), event.getTarget().hitVec.subtract( pos.getX(), pos.getY(), pos.getZ() ) ) | ||||||
|         } |             ? CableShapes.getModemShape( state ) : CableShapes.getCableShape( state ); | ||||||
|  |  | ||||||
|         if( type != PeripheralType.Cable && WorldUtil.isVecInsideInclusive( CableBounds.getModemBounds( state ), event.getTarget().hitVec.subtract( pos.getX(), pos.getY(), pos.getZ() ) ) ) |         WorldRenderer.drawShape( shape, pos.getX() - x, pos.getY() - y, pos.getZ() - z, 0.0F, 0.0F, 0.0F, 0.4F ); | ||||||
|         { |  | ||||||
|             RenderGlobal.drawSelectionBoundingBox( CableBounds.getModemBounds( state ), 0, 0, 0, 0.4f ); |  | ||||||
|         } |  | ||||||
|         else |  | ||||||
|         { |  | ||||||
|             int flags = 0; |  | ||||||
|  |  | ||||||
|             Tessellator tessellator = Tessellator.getInstance(); |  | ||||||
|             BufferBuilder buffer = tessellator.getBuffer(); |  | ||||||
|  |  | ||||||
|             for( EnumFacing facing : EnumFacing.VALUES ) |  | ||||||
|             { |  | ||||||
|                 if( BlockCable.doesConnectVisually( state, world, pos, facing ) ) |  | ||||||
|                 { |  | ||||||
|                     flags |= 1 << facing.ordinal(); |  | ||||||
|  |  | ||||||
|  |  | ||||||
|                     switch( facing.getAxis() ) |  | ||||||
|                     { |  | ||||||
|                         case X: |  | ||||||
|                         { |  | ||||||
|                             double offset = facing == EnumFacing.WEST ? -EXPAND : 1 + EXPAND; |  | ||||||
|                             double centre = facing == EnumFacing.WEST ? MIN : MAX; |  | ||||||
|  |  | ||||||
|                             buffer.begin( GL11.GL_LINE_STRIP, DefaultVertexFormats.POSITION ); |  | ||||||
|                             buffer.pos( offset, MIN, MIN ).endVertex(); |  | ||||||
|                             buffer.pos( offset, MAX, MIN ).endVertex(); |  | ||||||
|                             buffer.pos( offset, MAX, MAX ).endVertex(); |  | ||||||
|                             buffer.pos( offset, MIN, MAX ).endVertex(); |  | ||||||
|                             buffer.pos( offset, MIN, MIN ).endVertex(); |  | ||||||
|                             tessellator.draw(); |  | ||||||
|  |  | ||||||
|                             buffer.begin( GL11.GL_LINES, DefaultVertexFormats.POSITION ); |  | ||||||
|                             buffer.pos( offset, MIN, MIN ).endVertex(); |  | ||||||
|                             buffer.pos( centre, MIN, MIN ).endVertex(); |  | ||||||
|                             buffer.pos( offset, MAX, MIN ).endVertex(); |  | ||||||
|                             buffer.pos( centre, MAX, MIN ).endVertex(); |  | ||||||
|                             buffer.pos( offset, MAX, MAX ).endVertex(); |  | ||||||
|                             buffer.pos( centre, MAX, MAX ).endVertex(); |  | ||||||
|                             buffer.pos( offset, MIN, MAX ).endVertex(); |  | ||||||
|                             buffer.pos( centre, MIN, MAX ).endVertex(); |  | ||||||
|                             tessellator.draw(); |  | ||||||
|                             break; |  | ||||||
|                         } |  | ||||||
|                         case Y: |  | ||||||
|                         { |  | ||||||
|                             double offset = facing == EnumFacing.DOWN ? -EXPAND : 1 + EXPAND; |  | ||||||
|                             double centre = facing == EnumFacing.DOWN ? MIN : MAX; |  | ||||||
|  |  | ||||||
|                             buffer.begin( GL11.GL_LINE_STRIP, DefaultVertexFormats.POSITION ); |  | ||||||
|                             buffer.pos( MIN, offset, MIN ).endVertex(); |  | ||||||
|                             buffer.pos( MAX, offset, MIN ).endVertex(); |  | ||||||
|                             buffer.pos( MAX, offset, MAX ).endVertex(); |  | ||||||
|                             buffer.pos( MIN, offset, MAX ).endVertex(); |  | ||||||
|                             buffer.pos( MIN, offset, MIN ).endVertex(); |  | ||||||
|                             tessellator.draw(); |  | ||||||
|  |  | ||||||
|                             buffer.begin( GL11.GL_LINES, DefaultVertexFormats.POSITION ); |  | ||||||
|                             buffer.pos( MIN, offset, MIN ).endVertex(); |  | ||||||
|                             buffer.pos( MIN, centre, MIN ).endVertex(); |  | ||||||
|                             buffer.pos( MAX, offset, MIN ).endVertex(); |  | ||||||
|                             buffer.pos( MAX, centre, MIN ).endVertex(); |  | ||||||
|                             buffer.pos( MAX, offset, MAX ).endVertex(); |  | ||||||
|                             buffer.pos( MAX, centre, MAX ).endVertex(); |  | ||||||
|                             buffer.pos( MIN, offset, MAX ).endVertex(); |  | ||||||
|                             buffer.pos( MIN, centre, MAX ).endVertex(); |  | ||||||
|                             tessellator.draw(); |  | ||||||
|                             break; |  | ||||||
|                         } |  | ||||||
|                         case Z: |  | ||||||
|                         { |  | ||||||
|                             double offset = facing == EnumFacing.NORTH ? -EXPAND : 1 + EXPAND; |  | ||||||
|                             double centre = facing == EnumFacing.NORTH ? MIN : MAX; |  | ||||||
|  |  | ||||||
|                             buffer.begin( GL11.GL_LINE_STRIP, DefaultVertexFormats.POSITION ); |  | ||||||
|                             buffer.pos( MIN, MIN, offset ).endVertex(); |  | ||||||
|                             buffer.pos( MAX, MIN, offset ).endVertex(); |  | ||||||
|                             buffer.pos( MAX, MAX, offset ).endVertex(); |  | ||||||
|                             buffer.pos( MIN, MAX, offset ).endVertex(); |  | ||||||
|                             buffer.pos( MIN, MIN, offset ).endVertex(); |  | ||||||
|                             tessellator.draw(); |  | ||||||
|  |  | ||||||
|                             buffer.begin( GL11.GL_LINES, DefaultVertexFormats.POSITION ); |  | ||||||
|                             buffer.pos( MIN, MIN, offset ).endVertex(); |  | ||||||
|                             buffer.pos( MIN, MIN, centre ).endVertex(); |  | ||||||
|                             buffer.pos( MAX, MIN, offset ).endVertex(); |  | ||||||
|                             buffer.pos( MAX, MIN, centre ).endVertex(); |  | ||||||
|                             buffer.pos( MAX, MAX, offset ).endVertex(); |  | ||||||
|                             buffer.pos( MAX, MAX, centre ).endVertex(); |  | ||||||
|                             buffer.pos( MIN, MAX, offset ).endVertex(); |  | ||||||
|                             buffer.pos( MIN, MAX, centre ).endVertex(); |  | ||||||
|                             tessellator.draw(); |  | ||||||
|                             break; |  | ||||||
|                         } |  | ||||||
|                     } |  | ||||||
|                 } |  | ||||||
|             } |  | ||||||
|  |  | ||||||
|             buffer.begin( GL11.GL_LINES, DefaultVertexFormats.POSITION ); |  | ||||||
|  |  | ||||||
|             draw( buffer, flags, EnumFacing.WEST, EnumFacing.DOWN, EnumFacing.Axis.Z ); |  | ||||||
|             draw( buffer, flags, EnumFacing.WEST, EnumFacing.UP, EnumFacing.Axis.Z ); |  | ||||||
|             draw( buffer, flags, EnumFacing.EAST, EnumFacing.DOWN, EnumFacing.Axis.Z ); |  | ||||||
|             draw( buffer, flags, EnumFacing.EAST, EnumFacing.UP, EnumFacing.Axis.Z ); |  | ||||||
|  |  | ||||||
|             draw( buffer, flags, EnumFacing.WEST, EnumFacing.NORTH, EnumFacing.Axis.Y ); |  | ||||||
|             draw( buffer, flags, EnumFacing.WEST, EnumFacing.SOUTH, EnumFacing.Axis.Y ); |  | ||||||
|             draw( buffer, flags, EnumFacing.EAST, EnumFacing.NORTH, EnumFacing.Axis.Y ); |  | ||||||
|             draw( buffer, flags, EnumFacing.EAST, EnumFacing.SOUTH, EnumFacing.Axis.Y ); |  | ||||||
|  |  | ||||||
|             draw( buffer, flags, EnumFacing.DOWN, EnumFacing.NORTH, EnumFacing.Axis.X ); |  | ||||||
|             draw( buffer, flags, EnumFacing.DOWN, EnumFacing.SOUTH, EnumFacing.Axis.X ); |  | ||||||
|             draw( buffer, flags, EnumFacing.UP, EnumFacing.NORTH, EnumFacing.Axis.X ); |  | ||||||
|             draw( buffer, flags, EnumFacing.UP, EnumFacing.SOUTH, EnumFacing.Axis.X ); |  | ||||||
|  |  | ||||||
|             tessellator.draw(); |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         GlStateManager.popMatrix(); |         GlStateManager.popMatrix(); | ||||||
|  |         GlStateManager.matrixMode( GL11.GL_MODELVIEW ); | ||||||
|         GlStateManager.depthMask( true ); |         GlStateManager.depthMask( true ); | ||||||
|         GlStateManager.enableTexture2D(); |         GlStateManager.enableTexture2D(); | ||||||
|         GlStateManager.disableBlend(); |         GlStateManager.disableBlend(); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     private static void draw( BufferBuilder buffer, int flags, EnumFacing a, EnumFacing b, EnumFacing.Axis other ) |  | ||||||
|     { |  | ||||||
|         if( ((flags >> a.ordinal()) & 1) != ((flags >> b.ordinal()) & 1) ) return; |  | ||||||
|  |  | ||||||
|         double offA = a.getAxisDirection() == EnumFacing.AxisDirection.NEGATIVE ? MIN : MAX; |  | ||||||
|         double offB = b.getAxisDirection() == EnumFacing.AxisDirection.NEGATIVE ? MIN : MAX; |  | ||||||
|         switch( other ) |  | ||||||
|         { |  | ||||||
|             case X: |  | ||||||
|                 buffer.pos( MIN, offA, offB ).endVertex(); |  | ||||||
|                 buffer.pos( MAX, offA, offB ).endVertex(); |  | ||||||
|                 break; |  | ||||||
|             case Y: |  | ||||||
|                 buffer.pos( offA, MIN, offB ).endVertex(); |  | ||||||
|                 buffer.pos( offA, MAX, offB ).endVertex(); |  | ||||||
|                 break; |  | ||||||
|             case Z: |  | ||||||
|                 buffer.pos( offA, offB, MIN ).endVertex(); |  | ||||||
|                 buffer.pos( offA, offB, MAX ).endVertex(); |  | ||||||
|                 break; |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| } | } | ||||||
|   | |||||||
| @@ -7,46 +7,48 @@ | |||||||
| package dan200.computercraft.client.render; | package dan200.computercraft.client.render; | ||||||
|  |  | ||||||
| import dan200.computercraft.ComputerCraft; | import dan200.computercraft.ComputerCraft; | ||||||
| import dan200.computercraft.shared.peripheral.PeripheralType; |  | ||||||
| import dan200.computercraft.shared.peripheral.modem.wired.BlockCable; | import dan200.computercraft.shared.peripheral.modem.wired.BlockCable; | ||||||
| import dan200.computercraft.shared.peripheral.modem.wired.BlockCableModemVariant; | import dan200.computercraft.shared.peripheral.modem.wired.CableModemVariant; | ||||||
| import dan200.computercraft.shared.peripheral.modem.wired.CableBounds; | import dan200.computercraft.shared.peripheral.modem.wired.CableShapes; | ||||||
| import dan200.computercraft.shared.peripheral.modem.wired.TileCable; | import dan200.computercraft.shared.peripheral.modem.wired.TileCable; | ||||||
| import dan200.computercraft.shared.util.WorldUtil; |  | ||||||
| import net.minecraft.block.Block; | import net.minecraft.block.Block; | ||||||
| import net.minecraft.block.state.IBlockState; | import net.minecraft.block.state.IBlockState; | ||||||
| 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.RenderGlobal; |  | ||||||
| import net.minecraft.client.renderer.Tessellator; | import net.minecraft.client.renderer.Tessellator; | ||||||
| import net.minecraft.client.renderer.block.model.IBakedModel; | import net.minecraft.client.renderer.WorldRenderer; | ||||||
|  | import net.minecraft.client.renderer.model.IBakedModel; | ||||||
| import net.minecraft.client.renderer.texture.TextureAtlasSprite; | import net.minecraft.client.renderer.texture.TextureAtlasSprite; | ||||||
| import net.minecraft.client.renderer.tileentity.TileEntitySpecialRenderer; | import net.minecraft.client.renderer.tileentity.TileEntityRenderer; | ||||||
| import net.minecraft.client.renderer.vertex.DefaultVertexFormats; | import net.minecraft.client.renderer.vertex.DefaultVertexFormats; | ||||||
| import net.minecraft.util.BlockRenderLayer; | import net.minecraft.util.BlockRenderLayer; | ||||||
| import net.minecraft.util.math.BlockPos; | import net.minecraft.util.math.BlockPos; | ||||||
| import net.minecraft.util.math.RayTraceResult; | import net.minecraft.util.math.RayTraceResult; | ||||||
|  | import net.minecraft.util.math.shapes.VoxelShape; | ||||||
| import net.minecraft.world.World; | import net.minecraft.world.World; | ||||||
| import net.minecraftforge.client.ForgeHooksClient; | import net.minecraftforge.client.ForgeHooksClient; | ||||||
| import net.minecraftforge.client.MinecraftForgeClient; | import net.minecraftforge.client.MinecraftForgeClient; | ||||||
| import org.lwjgl.opengl.GL11; | import org.lwjgl.opengl.GL11; | ||||||
|  |  | ||||||
| import javax.annotation.Nonnull; | import javax.annotation.Nonnull; | ||||||
|  | import java.util.Random; | ||||||
|  |  | ||||||
| /** | /** | ||||||
|  * Render breaking animation only over part of a {@link TileCable}. |  * Render breaking animation only over part of a {@link TileCable}. | ||||||
|  */ |  */ | ||||||
| public class TileEntityCableRenderer extends TileEntitySpecialRenderer<TileCable> | public class TileEntityCableRenderer extends TileEntityRenderer<TileCable> | ||||||
| { | { | ||||||
|  |     private static final Random random = new Random(); | ||||||
|  |  | ||||||
|     @Override |     @Override | ||||||
|     public void render( @Nonnull TileCable te, double x, double y, double z, float partialTicks, int destroyStage, float alpha ) |     public void render( @Nonnull TileCable te, double x, double y, double z, float partialTicks, int destroyStage ) | ||||||
|     { |     { | ||||||
|         if( destroyStage < 0 ) return; |         if( destroyStage < 0 ) return; | ||||||
|  |  | ||||||
|         BlockPos pos = te.getPos(); |         BlockPos pos = te.getPos(); | ||||||
|  |  | ||||||
|         Minecraft mc = Minecraft.getMinecraft(); |         Minecraft mc = Minecraft.getInstance(); | ||||||
|  |  | ||||||
|         RayTraceResult hit = mc.objectMouseOver; |         RayTraceResult hit = mc.objectMouseOver; | ||||||
|         if( hit == null || !hit.getBlockPos().equals( pos ) ) return; |         if( hit == null || !hit.getBlockPos().equals( pos ) ) return; | ||||||
| @@ -58,18 +60,12 @@ public class TileEntityCableRenderer extends TileEntitySpecialRenderer<TileCable | |||||||
|         Block block = state.getBlock(); |         Block block = state.getBlock(); | ||||||
|         if( block != ComputerCraft.Blocks.cable ) return; |         if( block != ComputerCraft.Blocks.cable ) return; | ||||||
|  |  | ||||||
|         state = state.getActualState( world, pos ); |         VoxelShape shape = CableShapes.getModemShape( state ); | ||||||
|         if( te.getPeripheralType() != PeripheralType.Cable && WorldUtil.isVecInsideInclusive( CableBounds.getModemBounds( state ), hit.hitVec.subtract( pos.getX(), pos.getY(), pos.getZ() ) ) ) |         state = te.hasModem() && shape.getBoundingBox().grow( 0.02, 0.02, 0.02 ).contains( hit.hitVec.subtract( pos.getX(), pos.getY(), pos.getZ() ) ) | ||||||
|         { |             ? block.getDefaultState().with( BlockCable.MODEM, state.get( BlockCable.MODEM ) ) | ||||||
|             state = block.getDefaultState().withProperty( BlockCable.MODEM, state.getValue( BlockCable.MODEM ) ); |             : state.with( BlockCable.MODEM, CableModemVariant.None ); | ||||||
|         } |  | ||||||
|         else |  | ||||||
|         { |  | ||||||
|             state = state.withProperty( BlockCable.MODEM, BlockCableModemVariant.None ); |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         IBakedModel model = mc.getBlockRendererDispatcher().getModelForState( state ); |         IBakedModel model = mc.getBlockRendererDispatcher().getModelForState( state ); | ||||||
|         if( model == null ) return; |  | ||||||
|  |  | ||||||
|         preRenderDamagedBlocks(); |         preRenderDamagedBlocks(); | ||||||
|  |  | ||||||
| @@ -81,11 +77,11 @@ public class TileEntityCableRenderer extends TileEntitySpecialRenderer<TileCable | |||||||
|         ForgeHooksClient.setRenderLayer( block.getRenderLayer() ); |         ForgeHooksClient.setRenderLayer( block.getRenderLayer() ); | ||||||
|  |  | ||||||
|         // See BlockRendererDispatcher#renderBlockDamage |         // See BlockRendererDispatcher#renderBlockDamage | ||||||
|         TextureAtlasSprite breakingTexture = mc.getTextureMapBlocks().getAtlasSprite( "minecraft:blocks/destroy_stage_" + destroyStage ); |         TextureAtlasSprite breakingTexture = mc.getTextureMap().getSprite( DESTROY_STAGES[destroyStage] ); | ||||||
|         Minecraft.getMinecraft().getBlockRendererDispatcher().getBlockModelRenderer().renderModel( |         mc.getBlockRendererDispatcher().getBlockModelRenderer().renderModel( | ||||||
|             world, |             world, | ||||||
|             ForgeHooksClient.getDamageModel( model, breakingTexture, state, world, pos ), |             ForgeHooksClient.getDamageModel( model, breakingTexture, state, world, pos ), | ||||||
|             state, pos, buffer, true |             state, pos, buffer, true, random, state.getPositionRandom( pos ) | ||||||
|         ); |         ); | ||||||
|  |  | ||||||
|         ForgeHooksClient.setRenderLayer( BlockRenderLayer.SOLID ); |         ForgeHooksClient.setRenderLayer( BlockRenderLayer.SOLID ); | ||||||
| @@ -97,30 +93,30 @@ public class TileEntityCableRenderer extends TileEntitySpecialRenderer<TileCable | |||||||
|     } |     } | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * @see RenderGlobal#preRenderDamagedBlocks() |      * @see WorldRenderer#preRenderDamagedBlocks() | ||||||
|      */ |      */ | ||||||
|     private void preRenderDamagedBlocks() |     private void preRenderDamagedBlocks() | ||||||
|     { |     { | ||||||
|         GlStateManager.disableLighting(); |         GlStateManager.disableLighting(); | ||||||
|  |  | ||||||
|         GlStateManager.enableBlend(); |         GlStateManager.enableBlend(); | ||||||
|         GlStateManager.tryBlendFuncSeparate( GlStateManager.SourceFactor.DST_COLOR, GlStateManager.DestFactor.SRC_COLOR, GlStateManager.SourceFactor.ONE, GlStateManager.DestFactor.ZERO ); |         GlStateManager.blendFuncSeparate( GlStateManager.SourceFactor.DST_COLOR, GlStateManager.DestFactor.SRC_COLOR, GlStateManager.SourceFactor.ONE, GlStateManager.DestFactor.ZERO ); | ||||||
|         GlStateManager.enableBlend(); |         GlStateManager.enableBlend(); | ||||||
|         GlStateManager.color( 1.0F, 1.0F, 1.0F, 0.5F ); |         GlStateManager.color4f( 1.0F, 1.0F, 1.0F, 0.5F ); | ||||||
|         GlStateManager.doPolygonOffset( -3.0F, -3.0F ); |         GlStateManager.polygonOffset( -3.0F, -3.0F ); | ||||||
|         GlStateManager.enablePolygonOffset(); |         GlStateManager.enablePolygonOffset(); | ||||||
|         GlStateManager.alphaFunc( 516, 0.1F ); |         GlStateManager.alphaFunc( 516, 0.1F ); | ||||||
|         GlStateManager.enableAlpha(); |         GlStateManager.enableAlphaTest(); | ||||||
|         GlStateManager.pushMatrix(); |         GlStateManager.pushMatrix(); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * @see RenderGlobal#postRenderDamagedBlocks() |      * @see WorldRenderer#postRenderDamagedBlocks() | ||||||
|      */ |      */ | ||||||
|     private void postRenderDamagedBlocks() |     private void postRenderDamagedBlocks() | ||||||
|     { |     { | ||||||
|         GlStateManager.disableAlpha(); |         GlStateManager.disableAlphaTest(); | ||||||
|         GlStateManager.doPolygonOffset( 0.0F, 0.0F ); |         GlStateManager.polygonOffset( 0.0F, 0.0F ); | ||||||
|         GlStateManager.disablePolygonOffset(); |         GlStateManager.disablePolygonOffset(); | ||||||
|         GlStateManager.disablePolygonOffset(); |         GlStateManager.disablePolygonOffset(); | ||||||
|         GlStateManager.depthMask( true ); |         GlStateManager.depthMask( true ); | ||||||
|   | |||||||
| @@ -20,16 +20,16 @@ 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.TileEntityRenderer; | ||||||
| import net.minecraft.client.renderer.vertex.DefaultVertexFormats; | import net.minecraft.client.renderer.vertex.DefaultVertexFormats; | ||||||
| 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; | ||||||
|  |  | ||||||
| public class TileEntityMonitorRenderer extends TileEntitySpecialRenderer<TileMonitor> | public class TileEntityMonitorRenderer extends TileEntityRenderer<TileMonitor> | ||||||
| { | { | ||||||
|     @Override |     @Override | ||||||
|     public void render( TileMonitor tileEntity, double posX, double posY, double posZ, float f, int i, float f2 ) |     public void render( TileMonitor tileEntity, double posX, double posY, double posZ, float f, int i ) | ||||||
|     { |     { | ||||||
|         if( tileEntity != null ) |         if( tileEntity != null ) | ||||||
|         { |         { | ||||||
| @@ -73,19 +73,19 @@ public class TileEntityMonitorRenderer extends TileEntitySpecialRenderer<TileMon | |||||||
|         try |         try | ||||||
|         { |         { | ||||||
|             // Setup initial transform |             // Setup initial transform | ||||||
|             GlStateManager.translate( posX + 0.5, posY + 0.5, posZ + 0.5 ); |             GlStateManager.translated( posX + 0.5, posY + 0.5, posZ + 0.5 ); | ||||||
|             GlStateManager.rotate( -yaw, 0.0f, 1.0f, 0.0f ); |             GlStateManager.rotatef( -yaw, 0.0f, 1.0f, 0.0f ); | ||||||
|             GlStateManager.rotate( pitch, 1.0f, 0.0f, 0.0f ); |             GlStateManager.rotatef( pitch, 1.0f, 0.0f, 0.0f ); | ||||||
|             GlStateManager.translate( |             GlStateManager.translated( | ||||||
|                 -0.5 + TileMonitor.RENDER_BORDER + TileMonitor.RENDER_MARGIN, |                 -0.5 + TileMonitor.RENDER_BORDER + TileMonitor.RENDER_MARGIN, | ||||||
|                 origin.getHeight() - 0.5 - (TileMonitor.RENDER_BORDER + TileMonitor.RENDER_MARGIN), |                 origin.getHeight() - 0.5 - (TileMonitor.RENDER_BORDER + TileMonitor.RENDER_MARGIN) + 0, | ||||||
|                 0.5 |                 0.5 | ||||||
|             ); |             ); | ||||||
|             double xSize = origin.getWidth() - 2.0 * (TileMonitor.RENDER_MARGIN + TileMonitor.RENDER_BORDER); |             double xSize = origin.getWidth() - 2.0 * (TileMonitor.RENDER_MARGIN + TileMonitor.RENDER_BORDER); | ||||||
|             double ySize = origin.getHeight() - 2.0 * (TileMonitor.RENDER_MARGIN + TileMonitor.RENDER_BORDER); |             double ySize = origin.getHeight() - 2.0 * (TileMonitor.RENDER_MARGIN + TileMonitor.RENDER_BORDER); | ||||||
|  |  | ||||||
|             // Get renderers |             // Get renderers | ||||||
|             Minecraft mc = Minecraft.getMinecraft(); |             Minecraft mc = Minecraft.getInstance(); | ||||||
|             Tessellator tessellator = Tessellator.getInstance(); |             Tessellator tessellator = Tessellator.getInstance(); | ||||||
|             BufferBuilder renderer = tessellator.getBuffer(); |             BufferBuilder renderer = tessellator.getBuffer(); | ||||||
|  |  | ||||||
| @@ -94,9 +94,9 @@ public class TileEntityMonitorRenderer extends TileEntitySpecialRenderer<TileMon | |||||||
|  |  | ||||||
|             // Draw the contents |             // Draw the contents | ||||||
|             GlStateManager.depthMask( false ); |             GlStateManager.depthMask( false ); | ||||||
|             OpenGlHelper.setLightmapTextureCoords( OpenGlHelper.lightmapTexUnit, 0xFF, 0xFF ); |             OpenGlHelper.glMultiTexCoord2f( OpenGlHelper.GL_TEXTURE1, 0xFFFF, 0xFFFF ); | ||||||
|             GlStateManager.disableLighting(); |             GlStateManager.disableLighting(); | ||||||
|             mc.entityRenderer.disableLightmap(); |             mc.gameRenderer.disableLightmap(); | ||||||
|             try |             try | ||||||
|             { |             { | ||||||
|                 Terminal terminal = originTerminal.getTerminal(); |                 Terminal terminal = originTerminal.getTerminal(); | ||||||
| @@ -124,14 +124,14 @@ public class TileEntityMonitorRenderer extends TileEntitySpecialRenderer<TileMon | |||||||
|                     { |                     { | ||||||
|                         double xScale = xSize / (width * FixedWidthFontRenderer.FONT_WIDTH); |                         double xScale = xSize / (width * FixedWidthFontRenderer.FONT_WIDTH); | ||||||
|                         double yScale = ySize / (height * FixedWidthFontRenderer.FONT_HEIGHT); |                         double yScale = ySize / (height * FixedWidthFontRenderer.FONT_HEIGHT); | ||||||
|                         GlStateManager.scale( xScale, -yScale, 1.0 ); |                         GlStateManager.scaled( xScale, -yScale, 1.0 ); | ||||||
|  |  | ||||||
|                         // Draw background |                         // Draw background | ||||||
|                         mc.getTextureManager().bindTexture( FixedWidthFontRenderer.BACKGROUND ); |                         mc.getTextureManager().bindTexture( FixedWidthFontRenderer.BACKGROUND ); | ||||||
|                         if( redraw ) |                         if( redraw ) | ||||||
|                         { |                         { | ||||||
|                             // Build background display list |                             // Build background display list | ||||||
|                             GlStateManager.glNewList( originTerminal.renderDisplayLists[0], GL11.GL_COMPILE ); |                             GlStateManager.newList( originTerminal.renderDisplayLists[0], GL11.GL_COMPILE ); | ||||||
|                             try |                             try | ||||||
|                             { |                             { | ||||||
|                                 double marginXSize = TileMonitor.RENDER_MARGIN / xScale; |                                 double marginXSize = TileMonitor.RENDER_MARGIN / xScale; | ||||||
| @@ -142,10 +142,10 @@ public class TileEntityMonitorRenderer extends TileEntitySpecialRenderer<TileMon | |||||||
|                                 GlStateManager.pushMatrix(); |                                 GlStateManager.pushMatrix(); | ||||||
|                                 try |                                 try | ||||||
|                                 { |                                 { | ||||||
|                                     GlStateManager.scale( 1.0, marginSquash, 1.0 ); |                                     GlStateManager.scaled( 1.0, marginSquash, 1.0 ); | ||||||
|                                     GlStateManager.translate( 0.0, -marginYSize / marginSquash, 0.0 ); |                                     GlStateManager.translated( 0.0, -marginYSize / marginSquash, 0.0 ); | ||||||
|                                     fontRenderer.drawStringBackgroundPart( 0, 0, terminal.getBackgroundColourLine( 0 ), marginXSize, marginXSize, greyscale, palette ); |                                     fontRenderer.drawStringBackgroundPart( 0, 0, terminal.getBackgroundColourLine( 0 ), marginXSize, marginXSize, greyscale, palette ); | ||||||
|                                     GlStateManager.translate( 0.0, (marginYSize + height * FixedWidthFontRenderer.FONT_HEIGHT) / marginSquash, 0.0 ); |                                     GlStateManager.translated( 0.0, (marginYSize + height * FixedWidthFontRenderer.FONT_HEIGHT) / marginSquash, 0.0 ); | ||||||
|                                     fontRenderer.drawStringBackgroundPart( 0, 0, terminal.getBackgroundColourLine( height - 1 ), marginXSize, marginXSize, greyscale, palette ); |                                     fontRenderer.drawStringBackgroundPart( 0, 0, terminal.getBackgroundColourLine( height - 1 ), marginXSize, marginXSize, greyscale, palette ); | ||||||
|                                 } |                                 } | ||||||
|                                 finally |                                 finally | ||||||
| @@ -167,7 +167,7 @@ public class TileEntityMonitorRenderer extends TileEntitySpecialRenderer<TileMon | |||||||
|                             } |                             } | ||||||
|                             finally |                             finally | ||||||
|                             { |                             { | ||||||
|                                 GlStateManager.glEndList(); |                                 GlStateManager.endList(); | ||||||
|                             } |                             } | ||||||
|                         } |                         } | ||||||
|                         GlStateManager.callList( originTerminal.renderDisplayLists[0] ); |                         GlStateManager.callList( originTerminal.renderDisplayLists[0] ); | ||||||
| @@ -178,7 +178,7 @@ public class TileEntityMonitorRenderer extends TileEntitySpecialRenderer<TileMon | |||||||
|                         if( redraw ) |                         if( redraw ) | ||||||
|                         { |                         { | ||||||
|                             // Build text display list |                             // Build text display list | ||||||
|                             GlStateManager.glNewList( originTerminal.renderDisplayLists[1], GL11.GL_COMPILE ); |                             GlStateManager.newList( originTerminal.renderDisplayLists[1], GL11.GL_COMPILE ); | ||||||
|                             try |                             try | ||||||
|                             { |                             { | ||||||
|                                 // Lines |                                 // Lines | ||||||
| @@ -195,7 +195,7 @@ public class TileEntityMonitorRenderer extends TileEntitySpecialRenderer<TileMon | |||||||
|                             } |                             } | ||||||
|                             finally |                             finally | ||||||
|                             { |                             { | ||||||
|                                 GlStateManager.glEndList(); |                                 GlStateManager.endList(); | ||||||
|                             } |                             } | ||||||
|                         } |                         } | ||||||
|                         GlStateManager.callList( originTerminal.renderDisplayLists[1] ); |                         GlStateManager.callList( originTerminal.renderDisplayLists[1] ); | ||||||
| @@ -206,7 +206,7 @@ public class TileEntityMonitorRenderer extends TileEntitySpecialRenderer<TileMon | |||||||
|                         if( redraw ) |                         if( redraw ) | ||||||
|                         { |                         { | ||||||
|                             // Build cursor display list |                             // Build cursor display list | ||||||
|                             GlStateManager.glNewList( originTerminal.renderDisplayLists[2], GL11.GL_COMPILE ); |                             GlStateManager.newList( originTerminal.renderDisplayLists[2], GL11.GL_COMPILE ); | ||||||
|                             try |                             try | ||||||
|                             { |                             { | ||||||
|                                 // Cursor |                                 // Cursor | ||||||
| @@ -227,7 +227,7 @@ public class TileEntityMonitorRenderer extends TileEntitySpecialRenderer<TileMon | |||||||
|                             } |                             } | ||||||
|                             finally |                             finally | ||||||
|                             { |                             { | ||||||
|                                 GlStateManager.glEndList(); |                                 GlStateManager.endList(); | ||||||
|                             } |                             } | ||||||
|                         } |                         } | ||||||
|                         if( FrameInfo.getGlobalCursorBlink() ) |                         if( FrameInfo.getGlobalCursorBlink() ) | ||||||
| @@ -262,7 +262,7 @@ public class TileEntityMonitorRenderer extends TileEntitySpecialRenderer<TileMon | |||||||
|             finally |             finally | ||||||
|             { |             { | ||||||
|                 GlStateManager.depthMask( true ); |                 GlStateManager.depthMask( true ); | ||||||
|                 mc.entityRenderer.enableLightmap(); |                 mc.gameRenderer.enableLightmap(); | ||||||
|                 GlStateManager.enableLighting(); |                 GlStateManager.enableLighting(); | ||||||
|             } |             } | ||||||
|  |  | ||||||
| @@ -285,7 +285,7 @@ public class TileEntityMonitorRenderer extends TileEntitySpecialRenderer<TileMon | |||||||
|         } |         } | ||||||
|         finally |         finally | ||||||
|         { |         { | ||||||
|             GlStateManager.color( 1.0f, 1.0f, 1.0f, 1.0f ); |             GlStateManager.color4f( 1.0f, 1.0f, 1.0f, 1.0f ); | ||||||
|             GlStateManager.popMatrix(); |             GlStateManager.popMatrix(); | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
|   | |||||||
| @@ -10,20 +10,21 @@ import dan200.computercraft.api.turtle.ITurtleUpgrade; | |||||||
| import dan200.computercraft.api.turtle.TurtleSide; | import dan200.computercraft.api.turtle.TurtleSide; | ||||||
| import dan200.computercraft.shared.computer.core.ComputerFamily; | import dan200.computercraft.shared.computer.core.ComputerFamily; | ||||||
| import dan200.computercraft.shared.turtle.blocks.TileTurtle; | import dan200.computercraft.shared.turtle.blocks.TileTurtle; | ||||||
|  | import dan200.computercraft.shared.util.DirectionUtil; | ||||||
| import dan200.computercraft.shared.util.Holiday; | import dan200.computercraft.shared.util.Holiday; | ||||||
| import dan200.computercraft.shared.util.HolidayUtil; | import dan200.computercraft.shared.util.HolidayUtil; | ||||||
| import net.minecraft.block.state.IBlockState; | import net.minecraft.block.state.IBlockState; | ||||||
| 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.EntityRenderer; | import net.minecraft.client.renderer.GameRenderer; | ||||||
| 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.block.model.BakedQuad; | import net.minecraft.client.renderer.model.BakedQuad; | ||||||
| import net.minecraft.client.renderer.block.model.IBakedModel; | import net.minecraft.client.renderer.model.IBakedModel; | ||||||
| import net.minecraft.client.renderer.block.model.ModelManager; | import net.minecraft.client.renderer.model.ModelManager; | ||||||
| import net.minecraft.client.renderer.block.model.ModelResourceLocation; | import net.minecraft.client.renderer.model.ModelResourceLocation; | ||||||
| import net.minecraft.client.renderer.texture.TextureMap; | import net.minecraft.client.renderer.texture.TextureMap; | ||||||
| import net.minecraft.client.renderer.tileentity.TileEntitySpecialRenderer; | import net.minecraft.client.renderer.tileentity.TileEntityRenderer; | ||||||
| import net.minecraft.client.renderer.vertex.DefaultVertexFormats; | import net.minecraft.client.renderer.vertex.DefaultVertexFormats; | ||||||
| import net.minecraft.client.renderer.vertex.VertexFormat; | import net.minecraft.client.renderer.vertex.VertexFormat; | ||||||
| import net.minecraft.util.EnumFacing; | import net.minecraft.util.EnumFacing; | ||||||
| @@ -36,16 +37,17 @@ import org.lwjgl.opengl.GL11; | |||||||
|  |  | ||||||
| import javax.vecmath.Matrix4f; | import javax.vecmath.Matrix4f; | ||||||
| import java.util.List; | import java.util.List; | ||||||
|  | import java.util.Random; | ||||||
|  |  | ||||||
| public class TileEntityTurtleRenderer extends TileEntitySpecialRenderer<TileTurtle> | public class TileEntityTurtleRenderer extends TileEntityRenderer<TileTurtle> | ||||||
| { | { | ||||||
|     private static final ModelResourceLocation NORMAL_TURTLE_MODEL = new ModelResourceLocation( "computercraft:turtle", "inventory" ); |     private static final ModelResourceLocation NORMAL_TURTLE_MODEL = new ModelResourceLocation( "computercraft:turtle_normal", "inventory" ); | ||||||
|     private static final ModelResourceLocation ADVANCED_TURTLE_MODEL = new ModelResourceLocation( "computercraft:turtle_advanced", "inventory" ); |     private static final ModelResourceLocation ADVANCED_TURTLE_MODEL = new ModelResourceLocation( "computercraft:turtle_advanced", "inventory" ); | ||||||
|     private static final ModelResourceLocation COLOUR_TURTLE_MODEL = new ModelResourceLocation( "computercraft:turtle_white", "inventory" ); |     private static final ModelResourceLocation COLOUR_TURTLE_MODEL = new ModelResourceLocation( "computercraft:turtle_colour", "inventory" ); | ||||||
|     private static final ModelResourceLocation ELF_OVERLAY_MODEL = new ModelResourceLocation( "computercraft:turtle_elf_overlay", "inventory" ); |     private static final ModelResourceLocation ELF_OVERLAY_MODEL = new ModelResourceLocation( "computercraft:turtle_elf_overlay", "inventory" ); | ||||||
|  |  | ||||||
|     @Override |     @Override | ||||||
|     public void render( TileTurtle tileEntity, double posX, double posY, double posZ, float partialTicks, int breaking, float f2 ) |     public void render( TileTurtle tileEntity, double posX, double posY, double posZ, float partialTicks, int breaking ) | ||||||
|     { |     { | ||||||
|         if( tileEntity != null ) renderTurtleAt( tileEntity, posX, posY, posZ, partialTicks ); |         if( tileEntity != null ) renderTurtleAt( tileEntity, posX, posY, posZ, partialTicks ); | ||||||
|     } |     } | ||||||
| @@ -85,7 +87,7 @@ public class TileEntityTurtleRenderer extends TileEntitySpecialRenderer<TileTurt | |||||||
|         if( label != null && rendererDispatcher.cameraHitResult != null && turtle.getPos().equals( rendererDispatcher.cameraHitResult.getBlockPos() ) ) |         if( label != null && rendererDispatcher.cameraHitResult != null && turtle.getPos().equals( rendererDispatcher.cameraHitResult.getBlockPos() ) ) | ||||||
|         { |         { | ||||||
|             setLightmapDisabled( true ); |             setLightmapDisabled( true ); | ||||||
|             EntityRenderer.drawNameplate( |             GameRenderer.drawNameplate( | ||||||
|                 getFontRenderer(), label, |                 getFontRenderer(), label, | ||||||
|                 (float) posX + 0.5F, (float) posY + 1.2F, (float) posZ + 0.5F, 0, |                 (float) posX + 0.5F, (float) posY + 1.2F, (float) posZ + 0.5F, 0, | ||||||
|                 rendererDispatcher.entityYaw, rendererDispatcher.entityPitch, false, false |                 rendererDispatcher.entityYaw, rendererDispatcher.entityPitch, false, false | ||||||
| @@ -96,22 +98,21 @@ public class TileEntityTurtleRenderer extends TileEntitySpecialRenderer<TileTurt | |||||||
|         GlStateManager.pushMatrix(); |         GlStateManager.pushMatrix(); | ||||||
|         try |         try | ||||||
|         { |         { | ||||||
|             IBlockState state = turtle.getWorld().getBlockState( turtle.getPos() ); |             IBlockState state = turtle.getBlockState(); | ||||||
|             // Setup the transform |             // Setup the transform | ||||||
|             Vec3d offset = turtle.getRenderOffset( partialTicks ); |             Vec3d offset = turtle.getRenderOffset( partialTicks ); | ||||||
|             float yaw = turtle.getRenderYaw( partialTicks ); |             float yaw = turtle.getRenderYaw( partialTicks ); | ||||||
|             GlStateManager.translate( posX + offset.x, posY + offset.y, posZ + offset.z ); |             GlStateManager.translated( posX + offset.x, posY + offset.y, posZ + offset.z ); | ||||||
|  |  | ||||||
|             // Render the turtle |             // Render the turtle | ||||||
|             GlStateManager.translate( 0.5f, 0.5f, 0.5f ); |             GlStateManager.translatef( 0.5f, 0.5f, 0.5f ); | ||||||
|             GlStateManager.rotate( 180.0f - yaw, 0.0f, 1.0f, 0.0f ); |             GlStateManager.rotatef( 180.0f - yaw, 0.0f, 1.0f, 0.0f ); | ||||||
|             if( label != null && (label.equals( "Dinnerbone" ) || label.equals( "Grumm" )) ) |             if( label != null && (label.equals( "Dinnerbone" ) || label.equals( "Grumm" )) ) | ||||||
|             { |             { | ||||||
|                 // Flip the model and swap the cull face as winding order will have changed. |                 // Flip the model and swap the cull face as winding order will have changed. | ||||||
|                 GlStateManager.scale( 1.0f, -1.0f, 1.0f ); |                 GlStateManager.scalef( 1.0f, -1.0f, 1.0f ); | ||||||
|                 GlStateManager.cullFace( GlStateManager.CullFace.FRONT ); |                 GlStateManager.cullFace( GlStateManager.CullFace.FRONT ); | ||||||
|             } |             } | ||||||
|             GlStateManager.translate( -0.5f, -0.5f, -0.5f ); |             GlStateManager.translatef( -0.5f, -0.5f, -0.5f ); | ||||||
|             // Render the turtle |             // Render the turtle | ||||||
|             int colour = turtle.getColour(); |             int colour = turtle.getColour(); | ||||||
|             ComputerFamily family = turtle.getFamily(); |             ComputerFamily family = turtle.getFamily(); | ||||||
| @@ -151,7 +152,7 @@ public class TileEntityTurtleRenderer extends TileEntitySpecialRenderer<TileTurt | |||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     private static void renderUpgrade( IBlockState state, TileTurtle turtle, TurtleSide side, float f ) |     private void renderUpgrade( IBlockState state, TileTurtle turtle, TurtleSide side, float f ) | ||||||
|     { |     { | ||||||
|         ITurtleUpgrade upgrade = turtle.getUpgrade( side ); |         ITurtleUpgrade upgrade = turtle.getUpgrade( side ); | ||||||
|         if( upgrade != null ) |         if( upgrade != null ) | ||||||
| @@ -160,9 +161,9 @@ public class TileEntityTurtleRenderer extends TileEntitySpecialRenderer<TileTurt | |||||||
|             try |             try | ||||||
|             { |             { | ||||||
|                 float toolAngle = turtle.getToolRenderAngle( side, f ); |                 float toolAngle = turtle.getToolRenderAngle( side, f ); | ||||||
|                 GlStateManager.translate( 0.0f, 0.5f, 0.5f ); |                 GlStateManager.translatef( 0.0f, 0.5f, 0.5f ); | ||||||
|                 GlStateManager.rotate( -toolAngle, 1.0f, 0.0f, 0.0f ); |                 GlStateManager.rotatef( -toolAngle, 1.0f, 0.0f, 0.0f ); | ||||||
|                 GlStateManager.translate( 0.0f, -0.5f, -0.5f ); |                 GlStateManager.translatef( 0.0f, -0.5f, -0.5f ); | ||||||
|  |  | ||||||
|                 Pair<IBakedModel, Matrix4f> pair = upgrade.getModel( turtle.getAccess(), side ); |                 Pair<IBakedModel, Matrix4f> pair = upgrade.getModel( turtle.getAccess(), side ); | ||||||
|                 if( pair != null ) |                 if( pair != null ) | ||||||
| @@ -184,22 +185,22 @@ public class TileEntityTurtleRenderer extends TileEntitySpecialRenderer<TileTurt | |||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     private static void renderModel( IBlockState state, ModelResourceLocation modelLocation, int[] tints ) |     private void renderModel( IBlockState state, ModelResourceLocation modelLocation, int[] tints ) | ||||||
|     { |     { | ||||||
|         Minecraft mc = Minecraft.getMinecraft(); |         Minecraft mc = Minecraft.getInstance(); | ||||||
|         ModelManager modelManager = mc.getRenderItem().getItemModelMesher().getModelManager(); |         ModelManager modelManager = mc.getItemRenderer().getItemModelMesher().getModelManager(); | ||||||
|         renderModel( state, modelManager.getModel( modelLocation ), tints ); |         renderModel( state, modelManager.getModel( modelLocation ), tints ); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     private static void renderModel( IBlockState state, IBakedModel model, int[] tints ) |     private void renderModel( IBlockState state, IBakedModel model, int[] tints ) | ||||||
|     { |     { | ||||||
|         Minecraft mc = Minecraft.getMinecraft(); |         Random random = new Random( 0 ); | ||||||
|         Tessellator tessellator = Tessellator.getInstance(); |         Tessellator tessellator = Tessellator.getInstance(); | ||||||
|         mc.getTextureManager().bindTexture( TextureMap.LOCATION_BLOCKS_TEXTURE ); |         rendererDispatcher.textureManager.bindTexture( TextureMap.LOCATION_BLOCKS_TEXTURE ); | ||||||
|         renderQuads( tessellator, model.getQuads( state, null, 0 ), tints ); |         renderQuads( tessellator, model.getQuads( state, null, random ), tints ); | ||||||
|         for( EnumFacing facing : EnumFacing.VALUES ) |         for( EnumFacing facing : DirectionUtil.FACINGS ) | ||||||
|         { |         { | ||||||
|             renderQuads( tessellator, model.getQuads( state, facing, 0 ), tints ); |             renderQuads( tessellator, model.getQuads( state, facing, random ), tints ); | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  |  | ||||||
|   | |||||||
| @@ -6,26 +6,29 @@ | |||||||
|  |  | ||||||
| package dan200.computercraft.client.render; | package dan200.computercraft.client.render; | ||||||
|  |  | ||||||
| import com.google.common.collect.ImmutableMap; |  | ||||||
| import dan200.computercraft.ComputerCraft; | import dan200.computercraft.ComputerCraft; | ||||||
| import net.minecraft.client.renderer.block.model.IBakedModel; | import net.minecraft.client.renderer.model.IBakedModel; | ||||||
|  | import net.minecraft.client.renderer.model.IUnbakedModel; | ||||||
| import net.minecraft.client.renderer.texture.TextureAtlasSprite; | import net.minecraft.client.renderer.texture.TextureAtlasSprite; | ||||||
| import net.minecraft.client.renderer.vertex.VertexFormat; | import net.minecraft.client.renderer.vertex.VertexFormat; | ||||||
| import net.minecraft.client.resources.IResourceManager; | import net.minecraft.resources.IResourceManager; | ||||||
| import net.minecraft.util.ResourceLocation; | import net.minecraft.util.ResourceLocation; | ||||||
| import net.minecraftforge.client.model.ICustomModelLoader; | import net.minecraftforge.client.model.ICustomModelLoader; | ||||||
| import net.minecraftforge.client.model.IModel; |  | ||||||
| import net.minecraftforge.client.model.ModelLoaderRegistry; |  | ||||||
| import net.minecraftforge.common.model.IModelState; | import net.minecraftforge.common.model.IModelState; | ||||||
|  |  | ||||||
| import javax.annotation.Nonnull; | import javax.annotation.Nonnull; | ||||||
|  | import javax.annotation.Nullable; | ||||||
|  | import java.util.Arrays; | ||||||
|  | import java.util.Collection; | ||||||
|  | import java.util.Set; | ||||||
| import java.util.function.Function; | import java.util.function.Function; | ||||||
|  | import java.util.stream.Collectors; | ||||||
|  |  | ||||||
| public final class TurtleModelLoader implements ICustomModelLoader | public final class TurtleModelLoader implements ICustomModelLoader | ||||||
| { | { | ||||||
|     private static final ResourceLocation NORMAL_TURTLE_MODEL = new ResourceLocation( ComputerCraft.MOD_ID, "block/turtle" ); |     private static final ResourceLocation NORMAL_TURTLE_MODEL = new ResourceLocation( ComputerCraft.MOD_ID, "block/turtle_normal" ); | ||||||
|     private static final ResourceLocation ADVANCED_TURTLE_MODEL = new ResourceLocation( ComputerCraft.MOD_ID, "block/advanced_turtle" ); |     private static final ResourceLocation ADVANCED_TURTLE_MODEL = new ResourceLocation( ComputerCraft.MOD_ID, "block/turtle_advanced" ); | ||||||
|     private static final ResourceLocation COLOUR_TURTLE_MODEL = new ResourceLocation( ComputerCraft.MOD_ID, "block/turtle_white" ); |     private static final ResourceLocation COLOUR_TURTLE_MODEL = new ResourceLocation( ComputerCraft.MOD_ID, "block/turtle_colour" ); | ||||||
|  |  | ||||||
|     public static final TurtleModelLoader INSTANCE = new TurtleModelLoader(); |     public static final TurtleModelLoader INSTANCE = new TurtleModelLoader(); | ||||||
|  |  | ||||||
| @@ -42,80 +45,57 @@ public final class TurtleModelLoader implements ICustomModelLoader | |||||||
|     public boolean accepts( @Nonnull ResourceLocation name ) |     public boolean accepts( @Nonnull ResourceLocation name ) | ||||||
|     { |     { | ||||||
|         return name.getNamespace().equals( ComputerCraft.MOD_ID ) |         return name.getNamespace().equals( ComputerCraft.MOD_ID ) | ||||||
|             && (name.getPath().equals( "turtle" ) || name.getPath().equals( "turtle_advanced" )); |             && (name.getPath().equals( "item/turtle_normal" ) || name.getPath().equals( "item/turtle_advanced" )); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @Nonnull |     @Nonnull | ||||||
|     @Override |     @Override | ||||||
|     public IModel loadModel( @Nonnull ResourceLocation name ) throws Exception |     public IUnbakedModel loadModel( @Nonnull ResourceLocation name ) | ||||||
|     { |     { | ||||||
|         if( name.getNamespace().equals( ComputerCraft.MOD_ID ) ) |         if( name.getNamespace().equals( ComputerCraft.MOD_ID ) ) | ||||||
|         { |         { | ||||||
|             IModel colourModel = ModelLoaderRegistry.getModel( COLOUR_TURTLE_MODEL ); |  | ||||||
|             switch( name.getPath() ) |             switch( name.getPath() ) | ||||||
|             { |             { | ||||||
|                 case "turtle": |                 case "item/turtle_normal": | ||||||
|                     return new TurtleModel( ModelLoaderRegistry.getModel( NORMAL_TURTLE_MODEL ), colourModel ); |                     return new TurtleModel( NORMAL_TURTLE_MODEL ); | ||||||
|                 case "turtle_advanced": |                 case "item/turtle_advanced": | ||||||
|                     return new TurtleModel( ModelLoaderRegistry.getModel( ADVANCED_TURTLE_MODEL ), colourModel ); |                     return new TurtleModel( ADVANCED_TURTLE_MODEL ); | ||||||
|             } |             } | ||||||
|         } |         } | ||||||
|  |  | ||||||
|         throw new IllegalStateException( "Loader does not accept " + name ); |         throw new IllegalStateException( "Loader does not accept " + name ); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     private static final class TurtleModel implements IModel |     private static final class TurtleModel implements IUnbakedModel | ||||||
|     { |     { | ||||||
|         private final IModel family; |         private final ResourceLocation family; | ||||||
|         private final IModel colour; |  | ||||||
|  |  | ||||||
|         private TurtleModel( IModel family, IModel colour ) |         private TurtleModel( ResourceLocation family ) {this.family = family;} | ||||||
|  |  | ||||||
|  |         @Nonnull | ||||||
|  |         @Override | ||||||
|  |         public Collection<ResourceLocation> getDependencies() | ||||||
|         { |         { | ||||||
|             this.family = family; |             return Arrays.asList( family, COLOUR_TURTLE_MODEL ); | ||||||
|             this.colour = colour; |  | ||||||
|         } |         } | ||||||
|  |  | ||||||
|         @Nonnull |         @Nonnull | ||||||
|         @Override |         @Override | ||||||
|         public IBakedModel bake( @Nonnull IModelState state, @Nonnull VertexFormat format, @Nonnull Function<ResourceLocation, TextureAtlasSprite> function ) |         public Collection<ResourceLocation> getTextures( @Nonnull Function<ResourceLocation, IUnbakedModel> modelGetter, @Nonnull Set<String> missingTextureErrors ) | ||||||
|  |         { | ||||||
|  |             return getDependencies().stream() | ||||||
|  |                 .flatMap( x -> modelGetter.apply( x ).getTextures( modelGetter, missingTextureErrors ).stream() ) | ||||||
|  |                 .collect( Collectors.toSet() ); | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         @Nullable | ||||||
|  |         @Override | ||||||
|  |         public IBakedModel bake( @Nonnull Function<ResourceLocation, IUnbakedModel> modelGetter, @Nonnull Function<ResourceLocation, TextureAtlasSprite> spriteGetter, @Nonnull IModelState state, boolean uvlock, @Nonnull VertexFormat format ) | ||||||
|         { |         { | ||||||
|             return new TurtleSmartItemModel( |             return new TurtleSmartItemModel( | ||||||
|                 family.bake( state, format, function ), |                 modelGetter.apply( family ).bake( modelGetter, spriteGetter, state, uvlock, format ), | ||||||
|                 colour.bake( state, format, function ) |                 modelGetter.apply( COLOUR_TURTLE_MODEL ).bake( modelGetter, spriteGetter, state, uvlock, format ) | ||||||
|             ); |             ); | ||||||
|         } |         } | ||||||
|  |  | ||||||
|         private TurtleModel copy( IModel family, IModel colour ) |  | ||||||
|         { |  | ||||||
|             return this.family == family && this.colour == colour ? this : new TurtleModel( family, colour ); |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         @Nonnull |  | ||||||
|         @Override |  | ||||||
|         public IModel smoothLighting( boolean value ) |  | ||||||
|         { |  | ||||||
|             return copy( family.smoothLighting( value ), colour.smoothLighting( value ) ); |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         @Nonnull |  | ||||||
|         @Override |  | ||||||
|         public IModel gui3d( boolean value ) |  | ||||||
|         { |  | ||||||
|             return copy( family.gui3d( value ), colour.gui3d( value ) ); |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         @Nonnull |  | ||||||
|         @Override |  | ||||||
|         public IModel uvlock( boolean value ) |  | ||||||
|         { |  | ||||||
|             return copy( family.uvlock( value ), colour.uvlock( value ) ); |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         @Nonnull |  | ||||||
|         @Override |  | ||||||
|         public IModel retexture( ImmutableMap<String, String> textures ) |  | ||||||
|         { |  | ||||||
|             return copy( family.retexture( textures ), colour.retexture( textures ) ); |  | ||||||
|         } |  | ||||||
|     } |     } | ||||||
| } | } | ||||||
|   | |||||||
| @@ -7,19 +7,16 @@ | |||||||
| package dan200.computercraft.client.render; | package dan200.computercraft.client.render; | ||||||
|  |  | ||||||
| import net.minecraft.block.state.IBlockState; | import net.minecraft.block.state.IBlockState; | ||||||
| import net.minecraft.client.renderer.block.model.BakedQuad; | import net.minecraft.client.renderer.model.BakedQuad; | ||||||
| import net.minecraft.client.renderer.block.model.IBakedModel; | import net.minecraft.client.renderer.model.IBakedModel; | ||||||
| import net.minecraft.client.renderer.block.model.ItemCameraTransforms; | import net.minecraft.client.renderer.model.ItemCameraTransforms; | ||||||
| import net.minecraft.client.renderer.block.model.ItemOverrideList; | import net.minecraft.client.renderer.model.ItemOverrideList; | ||||||
| import net.minecraft.client.renderer.texture.TextureAtlasSprite; | import net.minecraft.client.renderer.texture.TextureAtlasSprite; | ||||||
| import net.minecraft.util.EnumFacing; | import net.minecraft.util.EnumFacing; | ||||||
|  |  | ||||||
| import javax.annotation.Nonnull; | import javax.annotation.Nonnull; | ||||||
| import javax.vecmath.Matrix4f; | import javax.vecmath.Matrix4f; | ||||||
| import java.util.ArrayList; | import java.util.*; | ||||||
| import java.util.EnumMap; |  | ||||||
| import java.util.List; |  | ||||||
| import java.util.Map; |  | ||||||
|  |  | ||||||
| public class TurtleMultiModel implements IBakedModel | public class TurtleMultiModel implements IBakedModel | ||||||
| { | { | ||||||
| @@ -47,7 +44,7 @@ public class TurtleMultiModel implements IBakedModel | |||||||
|  |  | ||||||
|     @Nonnull |     @Nonnull | ||||||
|     @Override |     @Override | ||||||
|     public List<BakedQuad> getQuads( IBlockState state, EnumFacing side, long rand ) |     public List<BakedQuad> getQuads( IBlockState state, EnumFacing side, @Nonnull Random rand ) | ||||||
|     { |     { | ||||||
|         if( side != null ) |         if( side != null ) | ||||||
|         { |         { | ||||||
| @@ -61,7 +58,7 @@ public class TurtleMultiModel implements IBakedModel | |||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     private List<BakedQuad> buildQuads( IBlockState state, EnumFacing side, long rand ) |     private List<BakedQuad> buildQuads( IBlockState state, EnumFacing side, Random rand ) | ||||||
|     { |     { | ||||||
|         ArrayList<BakedQuad> quads = new ArrayList<>(); |         ArrayList<BakedQuad> quads = new ArrayList<>(); | ||||||
|         ModelTransformer.transformQuadsTo( quads, m_baseModel.getQuads( state, side, rand ), m_generalTransform ); |         ModelTransformer.transformQuadsTo( quads, m_baseModel.getQuads( state, side, rand ), m_generalTransform ); | ||||||
| @@ -134,6 +131,6 @@ public class TurtleMultiModel implements IBakedModel | |||||||
|     @Override |     @Override | ||||||
|     public ItemOverrideList getOverrides() |     public ItemOverrideList getOverrides() | ||||||
|     { |     { | ||||||
|         return ItemOverrideList.NONE; |         return ItemOverrideList.EMPTY; | ||||||
|     } |     } | ||||||
| } | } | ||||||
|   | |||||||
| @@ -9,12 +9,12 @@ package dan200.computercraft.client.render; | |||||||
| import com.google.common.base.Objects; | import com.google.common.base.Objects; | ||||||
| import dan200.computercraft.api.turtle.ITurtleUpgrade; | import dan200.computercraft.api.turtle.ITurtleUpgrade; | ||||||
| import dan200.computercraft.api.turtle.TurtleSide; | import dan200.computercraft.api.turtle.TurtleSide; | ||||||
| import dan200.computercraft.shared.turtle.items.ItemTurtleBase; | import dan200.computercraft.shared.turtle.items.ItemTurtle; | ||||||
| import dan200.computercraft.shared.util.Holiday; | import dan200.computercraft.shared.util.Holiday; | ||||||
| import dan200.computercraft.shared.util.HolidayUtil; | import dan200.computercraft.shared.util.HolidayUtil; | ||||||
| import net.minecraft.block.state.IBlockState; | import net.minecraft.block.state.IBlockState; | ||||||
| import net.minecraft.client.Minecraft; | import net.minecraft.client.Minecraft; | ||||||
| import net.minecraft.client.renderer.block.model.*; | import net.minecraft.client.renderer.model.*; | ||||||
| import net.minecraft.client.renderer.texture.TextureAtlasSprite; | import net.minecraft.client.renderer.texture.TextureAtlasSprite; | ||||||
| import net.minecraft.entity.EntityLivingBase; | import net.minecraft.entity.EntityLivingBase; | ||||||
| import net.minecraft.item.ItemStack; | import net.minecraft.item.ItemStack; | ||||||
| @@ -26,9 +26,9 @@ import org.apache.commons.lang3.tuple.Pair; | |||||||
| import javax.annotation.Nonnull; | import javax.annotation.Nonnull; | ||||||
| import javax.annotation.Nullable; | import javax.annotation.Nullable; | ||||||
| import javax.vecmath.Matrix4f; | import javax.vecmath.Matrix4f; | ||||||
| import java.util.ArrayList; |  | ||||||
| import java.util.HashMap; | import java.util.HashMap; | ||||||
| import java.util.List; | import java.util.List; | ||||||
|  | import java.util.Random; | ||||||
|  |  | ||||||
| public class TurtleSmartItemModel implements IBakedModel | public class TurtleSmartItemModel implements IBakedModel | ||||||
| { | { | ||||||
| @@ -106,13 +106,13 @@ public class TurtleSmartItemModel implements IBakedModel | |||||||
|         this.colourModel = colourModel; |         this.colourModel = colourModel; | ||||||
|  |  | ||||||
|         m_cachedModels = new HashMap<>(); |         m_cachedModels = new HashMap<>(); | ||||||
|         m_overrides = new ItemOverrideList( new ArrayList<>() ) |         m_overrides = new ItemOverrideList() | ||||||
|         { |         { | ||||||
|             @Nonnull |             @Nonnull | ||||||
|             @Override |             @Override | ||||||
|             public IBakedModel handleItemState( @Nonnull IBakedModel originalModel, @Nonnull ItemStack stack, @Nullable World world, @Nullable EntityLivingBase entity ) |             public IBakedModel getModelWithOverrides( @Nonnull IBakedModel originalModel, @Nonnull ItemStack stack, @Nullable World world, @Nullable EntityLivingBase entity ) | ||||||
|             { |             { | ||||||
|                 ItemTurtleBase turtle = (ItemTurtleBase) stack.getItem(); |                 ItemTurtle turtle = (ItemTurtle) stack.getItem(); | ||||||
|                 int colour = turtle.getColour( stack ); |                 int colour = turtle.getColour( stack ); | ||||||
|                 ITurtleUpgrade leftUpgrade = turtle.getUpgrade( stack, TurtleSide.Left ); |                 ITurtleUpgrade leftUpgrade = turtle.getUpgrade( stack, TurtleSide.Left ); | ||||||
|                 ITurtleUpgrade rightUpgrade = turtle.getUpgrade( stack, TurtleSide.Right ); |                 ITurtleUpgrade rightUpgrade = turtle.getUpgrade( stack, TurtleSide.Right ); | ||||||
| @@ -138,8 +138,8 @@ public class TurtleSmartItemModel implements IBakedModel | |||||||
|  |  | ||||||
|     private IBakedModel buildModel( TurtleModelCombination combo ) |     private IBakedModel buildModel( TurtleModelCombination combo ) | ||||||
|     { |     { | ||||||
|         Minecraft mc = Minecraft.getMinecraft(); |         Minecraft mc = Minecraft.getInstance(); | ||||||
|         ModelManager modelManager = mc.getRenderItem().getItemModelMesher().getModelManager(); |         ModelManager modelManager = mc.getItemRenderer().getItemModelMesher().getModelManager(); | ||||||
|         ModelResourceLocation overlayModelLocation = TileEntityTurtleRenderer.getTurtleOverlayModel( combo.m_overlay, combo.m_christmas ); |         ModelResourceLocation overlayModelLocation = TileEntityTurtleRenderer.getTurtleOverlayModel( combo.m_overlay, combo.m_christmas ); | ||||||
|  |  | ||||||
|         IBakedModel baseModel = combo.m_colour ? colourModel : familyModel; |         IBakedModel baseModel = combo.m_colour ? colourModel : familyModel; | ||||||
| @@ -167,7 +167,7 @@ public class TurtleSmartItemModel implements IBakedModel | |||||||
|  |  | ||||||
|     @Nonnull |     @Nonnull | ||||||
|     @Override |     @Override | ||||||
|     public List<BakedQuad> getQuads( IBlockState state, EnumFacing facing, long rand ) |     public List<BakedQuad> getQuads( IBlockState state, EnumFacing facing, @Nonnull Random rand ) | ||||||
|     { |     { | ||||||
|         return familyModel.getQuads( state, facing, rand ); |         return familyModel.getQuads( state, facing, rand ); | ||||||
|     } |     } | ||||||
|   | |||||||
| @@ -12,6 +12,7 @@ import dan200.computercraft.ComputerCraft; | |||||||
| import java.net.Inet6Address; | import java.net.Inet6Address; | ||||||
| import java.net.InetAddress; | import java.net.InetAddress; | ||||||
| import java.util.ArrayList; | import java.util.ArrayList; | ||||||
|  | import java.util.Arrays; | ||||||
| import java.util.List; | import java.util.List; | ||||||
| import java.util.regex.Pattern; | import java.util.regex.Pattern; | ||||||
|  |  | ||||||
| @@ -50,6 +51,11 @@ public class AddressPredicate | |||||||
|     private final List<HostRange> ranges; |     private final List<HostRange> ranges; | ||||||
|  |  | ||||||
|     public AddressPredicate( String... filters ) |     public AddressPredicate( String... filters ) | ||||||
|  |     { | ||||||
|  |         this( Arrays.asList( filters ) ); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     public AddressPredicate( Iterable<? extends String> filters ) | ||||||
|     { |     { | ||||||
|         List<Pattern> wildcards = this.wildcards = new ArrayList<>(); |         List<Pattern> wildcards = this.wildcards = new ArrayList<>(); | ||||||
|         List<HostRange> ranges = this.ranges = new ArrayList<>(); |         List<HostRange> ranges = this.ranges = new ArrayList<>(); | ||||||
|   | |||||||
| @@ -23,7 +23,7 @@ public final class ApiFactories | |||||||
|     private static final Collection<ILuaAPIFactory> factories = new LinkedHashSet<>(); |     private static final Collection<ILuaAPIFactory> factories = new LinkedHashSet<>(); | ||||||
|     private static final Collection<ILuaAPIFactory> factoriesView = Collections.unmodifiableCollection( factories ); |     private static final Collection<ILuaAPIFactory> factoriesView = Collections.unmodifiableCollection( factories ); | ||||||
|  |  | ||||||
|     public static void register( @Nonnull ILuaAPIFactory factory ) |     public static synchronized void register( @Nonnull ILuaAPIFactory factory ) | ||||||
|     { |     { | ||||||
|         Preconditions.checkNotNull( factory, "provider cannot be null" ); |         Preconditions.checkNotNull( factory, "provider cannot be null" ); | ||||||
|         factories.add( factory ); |         factories.add( factory ); | ||||||
|   | |||||||
| @@ -8,7 +8,6 @@ package dan200.computercraft.core.computer; | |||||||
|  |  | ||||||
| import com.google.common.base.Objects; | import com.google.common.base.Objects; | ||||||
| import dan200.computercraft.api.lua.ILuaAPI; | import dan200.computercraft.api.lua.ILuaAPI; | ||||||
| import dan200.computercraft.api.peripheral.IPeripheral; |  | ||||||
| import dan200.computercraft.api.peripheral.IWorkMonitor; | import dan200.computercraft.api.peripheral.IWorkMonitor; | ||||||
| import dan200.computercraft.core.apis.IAPIEnvironment; | import dan200.computercraft.core.apis.IAPIEnvironment; | ||||||
| import dan200.computercraft.core.filesystem.FileSystem; | import dan200.computercraft.core.filesystem.FileSystem; | ||||||
| @@ -218,32 +217,4 @@ public class Computer | |||||||
|     { |     { | ||||||
|         executor.addApi( api ); |         executor.addApi( api ); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @Deprecated |  | ||||||
|     public IPeripheral getPeripheral( int side ) |  | ||||||
|     { |  | ||||||
|         return internalEnvironment.getPeripheral( side ); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     @Deprecated |  | ||||||
|     public void setPeripheral( int side, IPeripheral peripheral ) |  | ||||||
|     { |  | ||||||
|         internalEnvironment.setPeripheral( side, peripheral ); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     @Deprecated |  | ||||||
|     public void addAPI( dan200.computercraft.core.apis.ILuaAPI api ) |  | ||||||
|     { |  | ||||||
|         addApi( api ); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     @Deprecated |  | ||||||
|     @SuppressWarnings( "unused" ) |  | ||||||
|     public void advance( double dt ) |  | ||||||
|     { |  | ||||||
|         tick(); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     @Deprecated |  | ||||||
|     public static final String[] s_sideNames = IAPIEnvironment.SIDE_NAMES; |  | ||||||
| } | } | ||||||
|   | |||||||
| @@ -0,0 +1,275 @@ | |||||||
|  | /* | ||||||
|  |  * This file is part of ComputerCraft - http://www.computercraft.info | ||||||
|  |  * Copyright Daniel Ratcliffe, 2011-2019. Do not distribute without permission. | ||||||
|  |  * Send enquiries to dratcliffe@gmail.com | ||||||
|  |  */ | ||||||
|  |  | ||||||
|  | package dan200.computercraft.core.filesystem; | ||||||
|  |  | ||||||
|  | import com.google.common.cache.Cache; | ||||||
|  | import com.google.common.cache.CacheBuilder; | ||||||
|  | import com.google.common.io.ByteStreams; | ||||||
|  | import dan200.computercraft.api.filesystem.IMount; | ||||||
|  | import dan200.computercraft.core.apis.handles.ArrayByteChannel; | ||||||
|  | import net.minecraft.resources.IReloadableResourceManager; | ||||||
|  | import net.minecraft.resources.IResource; | ||||||
|  | import net.minecraft.resources.IResourceManager; | ||||||
|  | import net.minecraft.resources.IResourceManagerReloadListener; | ||||||
|  | import net.minecraft.util.ResourceLocation; | ||||||
|  | import net.minecraftforge.resource.IResourceType; | ||||||
|  | import net.minecraftforge.resource.ISelectiveResourceReloadListener; | ||||||
|  |  | ||||||
|  | import javax.annotation.Nonnull; | ||||||
|  | import javax.annotation.Nullable; | ||||||
|  | import java.io.FileNotFoundException; | ||||||
|  | import java.io.IOException; | ||||||
|  | import java.io.InputStream; | ||||||
|  | import java.nio.channels.Channels; | ||||||
|  | import java.nio.channels.ReadableByteChannel; | ||||||
|  | import java.util.*; | ||||||
|  | import java.util.concurrent.TimeUnit; | ||||||
|  | import java.util.function.Predicate; | ||||||
|  |  | ||||||
|  | public class ResourceMount implements IMount | ||||||
|  | { | ||||||
|  |     /** | ||||||
|  |      * Only cache files smaller than 1MiB. | ||||||
|  |      */ | ||||||
|  |     private static final int MAX_CACHED_SIZE = 1 << 20; | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * Limit the entire cache to 64MiB. | ||||||
|  |      */ | ||||||
|  |     private static final int MAX_CACHE_SIZE = 64 << 20; | ||||||
|  |  | ||||||
|  |     private static final byte[] TEMP_BUFFER = new byte[8192]; | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * We maintain a cache of the contents of all files in the mount. This allows us to allow | ||||||
|  |      * seeking within ROM files, and reduces the amount we need to access disk for computer startup. | ||||||
|  |      */ | ||||||
|  |     private static final Cache<FileEntry, byte[]> CONTENTS_CACHE = CacheBuilder.newBuilder() | ||||||
|  |         .concurrencyLevel( 4 ) | ||||||
|  |         .expireAfterAccess( 60, TimeUnit.SECONDS ) | ||||||
|  |         .maximumWeight( MAX_CACHE_SIZE ) | ||||||
|  |         .weakKeys() | ||||||
|  |         .<FileEntry, byte[]>weigher( ( k, v ) -> v.length ) | ||||||
|  |         .build(); | ||||||
|  |  | ||||||
|  |     private final String namespace; | ||||||
|  |     private final String subPath; | ||||||
|  |     private final IReloadableResourceManager manager; | ||||||
|  |  | ||||||
|  |     @Nullable | ||||||
|  |     private FileEntry root; | ||||||
|  |  | ||||||
|  |     public ResourceMount( String namespace, String subPath, IReloadableResourceManager manager ) | ||||||
|  |     { | ||||||
|  |         this.namespace = namespace; | ||||||
|  |         this.subPath = subPath; | ||||||
|  |         this.manager = manager; | ||||||
|  |  | ||||||
|  |         Listener.INSTANCE.add( manager, this ); | ||||||
|  |         if( root == null ) load(); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     private void load() | ||||||
|  |     { | ||||||
|  |         boolean hasAny = false; | ||||||
|  |         FileEntry newRoot = new FileEntry( new ResourceLocation( namespace, subPath ) ); | ||||||
|  |         for( ResourceLocation file : manager.getAllResourceLocations( subPath, s -> true ) ) | ||||||
|  |         { | ||||||
|  |             if( !file.getNamespace().equals( namespace ) ) continue; | ||||||
|  |  | ||||||
|  |             String localPath = FileSystem.toLocal( file.getPath(), subPath ); | ||||||
|  |             create( newRoot, localPath ); | ||||||
|  |             hasAny = true; | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         root = hasAny ? newRoot : null; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     private FileEntry get( String path ) | ||||||
|  |     { | ||||||
|  |         FileEntry lastEntry = root; | ||||||
|  |         int lastIndex = 0; | ||||||
|  |  | ||||||
|  |         while( lastEntry != null && lastIndex < path.length() ) | ||||||
|  |         { | ||||||
|  |             int nextIndex = path.indexOf( '/', lastIndex ); | ||||||
|  |             if( nextIndex < 0 ) nextIndex = path.length(); | ||||||
|  |  | ||||||
|  |             lastEntry = lastEntry.children == null ? null : lastEntry.children.get( path.substring( lastIndex, nextIndex ) ); | ||||||
|  |             lastIndex = nextIndex + 1; | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         return lastEntry; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     private void create( FileEntry lastEntry, String path ) | ||||||
|  |     { | ||||||
|  |         int lastIndex = 0; | ||||||
|  |         while( lastIndex < path.length() ) | ||||||
|  |         { | ||||||
|  |             int nextIndex = path.indexOf( '/', lastIndex ); | ||||||
|  |             if( nextIndex < 0 ) nextIndex = path.length(); | ||||||
|  |  | ||||||
|  |             String part = path.substring( lastIndex, nextIndex ); | ||||||
|  |             if( lastEntry.children == null ) lastEntry.children = new HashMap<>(); | ||||||
|  |  | ||||||
|  |             FileEntry nextEntry = lastEntry.children.get( part ); | ||||||
|  |             if( nextEntry == null ) | ||||||
|  |             { | ||||||
|  |                 lastEntry.children.put( part, nextEntry = new FileEntry( new ResourceLocation( namespace, subPath + "/" + path ) ) ); | ||||||
|  |             } | ||||||
|  |  | ||||||
|  |             lastEntry = nextEntry; | ||||||
|  |             lastIndex = nextIndex + 1; | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     @Override | ||||||
|  |     public boolean exists( @Nonnull String path ) | ||||||
|  |     { | ||||||
|  |         return get( path ) != null; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     @Override | ||||||
|  |     public boolean isDirectory( @Nonnull String path ) | ||||||
|  |     { | ||||||
|  |         FileEntry file = get( path ); | ||||||
|  |         return file != null && file.isDirectory(); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     @Override | ||||||
|  |     public void list( @Nonnull String path, @Nonnull List<String> contents ) throws IOException | ||||||
|  |     { | ||||||
|  |         FileEntry file = get( path ); | ||||||
|  |         if( file == null || !file.isDirectory() ) throw new IOException( "/" + path + ": Not a directory" ); | ||||||
|  |  | ||||||
|  |         file.list( contents ); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     @Override | ||||||
|  |     public long getSize( @Nonnull String path ) throws IOException | ||||||
|  |     { | ||||||
|  |         FileEntry file = get( path ); | ||||||
|  |         if( file != null ) | ||||||
|  |         { | ||||||
|  |             if( file.size != -1 ) return file.size; | ||||||
|  |             if( file.isDirectory() ) return file.size = 0; | ||||||
|  |  | ||||||
|  |             byte[] contents = CONTENTS_CACHE.getIfPresent( file ); | ||||||
|  |             if( contents != null ) return file.size = contents.length; | ||||||
|  |  | ||||||
|  |             try | ||||||
|  |             { | ||||||
|  |                 IResource resource = manager.getResource( file.identifier ); | ||||||
|  |                 InputStream s = resource.getInputStream(); | ||||||
|  |                 int total = 0, read = 0; | ||||||
|  |                 do | ||||||
|  |                 { | ||||||
|  |                     total += read; | ||||||
|  |                     read = s.read( TEMP_BUFFER ); | ||||||
|  |                 } while( read > 0 ); | ||||||
|  |  | ||||||
|  |                 return file.size = total; | ||||||
|  |             } | ||||||
|  |             catch( IOException e ) | ||||||
|  |             { | ||||||
|  |                 return file.size = 0; | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         throw new IOException( "/" + path + ": No such file" ); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     @Nonnull | ||||||
|  |     @Override | ||||||
|  |     @Deprecated | ||||||
|  |     public InputStream openForRead( @Nonnull String path ) throws IOException | ||||||
|  |     { | ||||||
|  |         return Channels.newInputStream( openChannelForRead( path ) ); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     @Nonnull | ||||||
|  |     @Override | ||||||
|  |     public ReadableByteChannel openChannelForRead( @Nonnull String path ) throws IOException | ||||||
|  |     { | ||||||
|  |         FileEntry file = get( path ); | ||||||
|  |         if( file != null && !file.isDirectory() ) | ||||||
|  |         { | ||||||
|  |             byte[] contents = CONTENTS_CACHE.getIfPresent( file ); | ||||||
|  |             if( contents != null ) return new ArrayByteChannel( contents ); | ||||||
|  |  | ||||||
|  |             try( InputStream stream = manager.getResource( file.identifier ).getInputStream() ) | ||||||
|  |             { | ||||||
|  |                 if( stream.available() > MAX_CACHED_SIZE ) return Channels.newChannel( stream ); | ||||||
|  |  | ||||||
|  |                 contents = ByteStreams.toByteArray( stream ); | ||||||
|  |                 CONTENTS_CACHE.put( file, contents ); | ||||||
|  |                 return new ArrayByteChannel( contents ); | ||||||
|  |             } | ||||||
|  |             catch( FileNotFoundException ignored ) | ||||||
|  |             { | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         throw new IOException( "/" + path + ": No such file" ); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     private static class FileEntry | ||||||
|  |     { | ||||||
|  |         final ResourceLocation identifier; | ||||||
|  |         Map<String, FileEntry> children; | ||||||
|  |         long size = -1; | ||||||
|  |  | ||||||
|  |         FileEntry( ResourceLocation identifier ) | ||||||
|  |         { | ||||||
|  |             this.identifier = identifier; | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         boolean isDirectory() | ||||||
|  |         { | ||||||
|  |             return children != null; | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         void list( List<String> contents ) | ||||||
|  |         { | ||||||
|  |             if( children != null ) contents.addAll( children.keySet() ); | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * A {@link IResourceManagerReloadListener} which reloads any associated mounts. | ||||||
|  |      * | ||||||
|  |      * While people should really be keeping a permanent reference to this, some people construct it every | ||||||
|  |      * method call, so let's make this as small as possible. | ||||||
|  |      */ | ||||||
|  |     static class Listener implements ISelectiveResourceReloadListener | ||||||
|  |     { | ||||||
|  |         private static final Listener INSTANCE = new Listener(); | ||||||
|  |  | ||||||
|  |         private final Set<ResourceMount> mounts = Collections.newSetFromMap( new WeakHashMap<>() ); | ||||||
|  |         private final Set<IReloadableResourceManager> managers = Collections.newSetFromMap( new WeakHashMap<>() ); | ||||||
|  |  | ||||||
|  |         @Override | ||||||
|  |         public void onResourceManagerReload( @Nonnull IResourceManager manager ) | ||||||
|  |         { | ||||||
|  |             // FIXME: Remove this. We need this patch in order to prevent trying to load ReloadRequirements. | ||||||
|  |             onResourceManagerReload( manager, x -> true ); | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         @Override | ||||||
|  |         public synchronized void onResourceManagerReload( @Nonnull IResourceManager manager, @Nonnull Predicate<IResourceType> predicate ) | ||||||
|  |         { | ||||||
|  |             for( ResourceMount mount : mounts ) mount.load(); | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         synchronized void add( IReloadableResourceManager manager, ResourceMount mount ) | ||||||
|  |         { | ||||||
|  |             if( managers.add( manager ) ) manager.addReloadListener( this ); | ||||||
|  |             mounts.add( mount ); | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | } | ||||||
| @@ -336,16 +336,16 @@ public class Terminal | |||||||
|  |  | ||||||
|     public synchronized NBTTagCompound writeToNBT( NBTTagCompound nbt ) |     public synchronized NBTTagCompound writeToNBT( NBTTagCompound nbt ) | ||||||
|     { |     { | ||||||
|         nbt.setInteger( "term_cursorX", m_cursorX ); |         nbt.putInt( "term_cursorX", m_cursorX ); | ||||||
|         nbt.setInteger( "term_cursorY", m_cursorY ); |         nbt.putInt( "term_cursorY", m_cursorY ); | ||||||
|         nbt.setBoolean( "term_cursorBlink", m_cursorBlink ); |         nbt.putBoolean( "term_cursorBlink", m_cursorBlink ); | ||||||
|         nbt.setInteger( "term_textColour", m_cursorColour ); |         nbt.putInt( "term_textColour", m_cursorColour ); | ||||||
|         nbt.setInteger( "term_bgColour", m_cursorBackgroundColour ); |         nbt.putInt( "term_bgColour", m_cursorBackgroundColour ); | ||||||
|         for( int n = 0; n < m_height; n++ ) |         for( int n = 0; n < m_height; n++ ) | ||||||
|         { |         { | ||||||
|             nbt.setString( "term_text_" + n, m_text[n].toString() ); |             nbt.putString( "term_text_" + n, m_text[n].toString() ); | ||||||
|             nbt.setString( "term_textColour_" + n, m_textColour[n].toString() ); |             nbt.putString( "term_textColour_" + n, m_textColour[n].toString() ); | ||||||
|             nbt.setString( "term_textBgColour_" + n, m_backgroundColour[n].toString() ); |             nbt.putString( "term_textBgColour_" + n, m_backgroundColour[n].toString() ); | ||||||
|         } |         } | ||||||
|         if( m_palette != null ) |         if( m_palette != null ) | ||||||
|         { |         { | ||||||
| @@ -356,26 +356,26 @@ public class Terminal | |||||||
|  |  | ||||||
|     public synchronized void readFromNBT( NBTTagCompound nbt ) |     public synchronized void readFromNBT( NBTTagCompound nbt ) | ||||||
|     { |     { | ||||||
|         m_cursorX = nbt.getInteger( "term_cursorX" ); |         m_cursorX = nbt.getInt( "term_cursorX" ); | ||||||
|         m_cursorY = nbt.getInteger( "term_cursorY" ); |         m_cursorY = nbt.getInt( "term_cursorY" ); | ||||||
|         m_cursorBlink = nbt.getBoolean( "term_cursorBlink" ); |         m_cursorBlink = nbt.getBoolean( "term_cursorBlink" ); | ||||||
|         m_cursorColour = nbt.getInteger( "term_textColour" ); |         m_cursorColour = nbt.getInt( "term_textColour" ); | ||||||
|         m_cursorBackgroundColour = nbt.getInteger( "term_bgColour" ); |         m_cursorBackgroundColour = nbt.getInt( "term_bgColour" ); | ||||||
|  |  | ||||||
|         for( int n = 0; n < m_height; n++ ) |         for( int n = 0; n < m_height; n++ ) | ||||||
|         { |         { | ||||||
|             m_text[n].fill( ' ' ); |             m_text[n].fill( ' ' ); | ||||||
|             if( nbt.hasKey( "term_text_" + n ) ) |             if( nbt.contains( "term_text_" + n ) ) | ||||||
|             { |             { | ||||||
|                 m_text[n].write( nbt.getString( "term_text_" + n ) ); |                 m_text[n].write( nbt.getString( "term_text_" + n ) ); | ||||||
|             } |             } | ||||||
|             m_textColour[n].fill( base16.charAt( m_cursorColour ) ); |             m_textColour[n].fill( base16.charAt( m_cursorColour ) ); | ||||||
|             if( nbt.hasKey( "term_textColour_" + n ) ) |             if( nbt.contains( "term_textColour_" + n ) ) | ||||||
|             { |             { | ||||||
|                 m_textColour[n].write( nbt.getString( "term_textColour_" + n ) ); |                 m_textColour[n].write( nbt.getString( "term_textColour_" + n ) ); | ||||||
|             } |             } | ||||||
|             m_backgroundColour[n].fill( base16.charAt( m_cursorBackgroundColour ) ); |             m_backgroundColour[n].fill( base16.charAt( m_cursorBackgroundColour ) ); | ||||||
|             if( nbt.hasKey( "term_textBgColour_" + n ) ) |             if( nbt.contains( "term_textBgColour_" + n ) ) | ||||||
|             { |             { | ||||||
|                 m_backgroundColour[n].write( nbt.getString( "term_textBgColour_" + n ) ); |                 m_backgroundColour[n].write( nbt.getString( "term_textBgColour_" + n ) ); | ||||||
|             } |             } | ||||||
|   | |||||||
| @@ -10,11 +10,6 @@ import dan200.computercraft.core.computer.Computer; | |||||||
|  |  | ||||||
| public interface Tracker | public interface Tracker | ||||||
| { | { | ||||||
|     @Deprecated |  | ||||||
|     default void addTiming( Computer computer, long time ) |  | ||||||
|     { |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * Report how long a task executed on the computer thread took. |      * Report how long a task executed on the computer thread took. | ||||||
|      * |      * | ||||||
| @@ -25,8 +20,6 @@ public interface Tracker | |||||||
|      */ |      */ | ||||||
|     default void addTaskTiming( Computer computer, long time ) |     default void addTaskTiming( Computer computer, long time ) | ||||||
|     { |     { | ||||||
|         //noinspection deprecation |  | ||||||
|         addTiming( computer, time ); |  | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|   | |||||||
| @@ -6,8 +6,6 @@ | |||||||
|  |  | ||||||
| package dan200.computercraft.core.tracking; | package dan200.computercraft.core.tracking; | ||||||
|  |  | ||||||
| import dan200.computercraft.shared.util.StringUtil; |  | ||||||
|  |  | ||||||
| import java.util.Collections; | import java.util.Collections; | ||||||
| import java.util.HashMap; | import java.util.HashMap; | ||||||
| import java.util.Map; | import java.util.Map; | ||||||
| @@ -15,31 +13,29 @@ import java.util.function.LongFunction; | |||||||
|  |  | ||||||
| public final class TrackingField | public final class TrackingField | ||||||
| { | { | ||||||
|     public static final String TRANSLATE_PREFIX = "tracking_field.computercraft."; |  | ||||||
|  |  | ||||||
|     private static final Map<String, TrackingField> fields = new HashMap<>(); |     private static final Map<String, TrackingField> fields = new HashMap<>(); | ||||||
|  |  | ||||||
|     public static final TrackingField TASKS = TrackingField.of( "tasks", "Tasks", x -> String.format( "%4d", x ) ); |     public static final TrackingField TASKS = TrackingField.of( "tasks", x -> String.format( "%4d", x ) ); | ||||||
|     public static final TrackingField TOTAL_TIME = TrackingField.of( "total", "Total time", x -> String.format( "%7.1fms", x / 1e6 ) ); |     public static final TrackingField TOTAL_TIME = TrackingField.of( "total", x -> String.format( "%7.1fms", x / 1e6 ) ); | ||||||
|     public static final TrackingField AVERAGE_TIME = TrackingField.of( "average", "Average time", x -> String.format( "%4.1fms", x / 1e6 ) ); |     public static final TrackingField AVERAGE_TIME = TrackingField.of( "average", x -> String.format( "%4.1fms", x / 1e6 ) ); | ||||||
|     public static final TrackingField MAX_TIME = TrackingField.of( "max", "Max time", x -> String.format( "%5.1fms", x / 1e6 ) ); |     public static final TrackingField MAX_TIME = TrackingField.of( "max", x -> String.format( "%5.1fms", x / 1e6 ) ); | ||||||
|  |  | ||||||
|     public static final TrackingField SERVER_COUNT = TrackingField.of( "server_count", "Server task count", x -> String.format( "%4d", x ) ); |     public static final TrackingField SERVER_COUNT = TrackingField.of( "server_count", x -> String.format( "%4d", x ) ); | ||||||
|     public static final TrackingField SERVER_TIME = TrackingField.of( "server_time", "Server task time", x -> String.format( "%7.1fms", x / 1e6 ) ); |     public static final TrackingField SERVER_TIME = TrackingField.of( "server_time", x -> String.format( "%7.1fms", x / 1e6 ) ); | ||||||
|  |  | ||||||
|     public static final TrackingField PERIPHERAL_OPS = TrackingField.of( "peripheral", "Peripheral calls", TrackingField::formatDefault ); |     public static final TrackingField PERIPHERAL_OPS = TrackingField.of( "peripheral", TrackingField::formatDefault ); | ||||||
|     public static final TrackingField FS_OPS = TrackingField.of( "fs", "Filesystem operations", TrackingField::formatDefault ); |     public static final TrackingField FS_OPS = TrackingField.of( "fs", TrackingField::formatDefault ); | ||||||
|     public static final TrackingField TURTLE_OPS = TrackingField.of( "turtle", "Turtle operations", TrackingField::formatDefault ); |     public static final TrackingField TURTLE_OPS = TrackingField.of( "turtle", TrackingField::formatDefault ); | ||||||
|  |  | ||||||
|     public static final TrackingField HTTP_REQUESTS = TrackingField.of( "http", "HTTP requests", TrackingField::formatDefault ); |     public static final TrackingField HTTP_REQUESTS = TrackingField.of( "http", TrackingField::formatDefault ); | ||||||
|     public static final TrackingField HTTP_UPLOAD = TrackingField.of( "http_upload", "HTTP upload", TrackingField::formatBytes ); |     public static final TrackingField HTTP_UPLOAD = TrackingField.of( "http_upload", TrackingField::formatBytes ); | ||||||
|     public static final TrackingField HTTP_DOWNLOAD = TrackingField.of( "http_download", "HTTP download", TrackingField::formatBytes ); |     public static final TrackingField HTTP_DOWNLOAD = TrackingField.of( "http_download", TrackingField::formatBytes ); | ||||||
|  |  | ||||||
|     public static final TrackingField WEBSOCKET_INCOMING = TrackingField.of( "websocket_incoming", "Websocket incoming", TrackingField::formatBytes ); |     public static final TrackingField WEBSOCKET_INCOMING = TrackingField.of( "websocket_incoming", TrackingField::formatBytes ); | ||||||
|     public static final TrackingField WEBSOCKET_OUTGOING = TrackingField.of( "websocket_outgoing", "Websocket outgoing", TrackingField::formatBytes ); |     public static final TrackingField WEBSOCKET_OUTGOING = TrackingField.of( "websocket_outgoing", TrackingField::formatBytes ); | ||||||
|  |  | ||||||
|     public static final TrackingField COROUTINES_CREATED = TrackingField.of( "coroutines_created", "Coroutines created", x -> String.format( "%4d", x ) ); |     public static final TrackingField COROUTINES_CREATED = TrackingField.of( "coroutines_created", x -> String.format( "%4d", x ) ); | ||||||
|     public static final TrackingField COROUTINES_DISPOSED = TrackingField.of( "coroutines_dead", "Coroutines disposed", x -> String.format( "%4d", x ) ); |     public static final TrackingField COROUTINES_DISPOSED = TrackingField.of( "coroutines_dead", x -> String.format( "%4d", x ) ); | ||||||
|  |  | ||||||
|     private final String id; |     private final String id; | ||||||
|     private final String translationKey; |     private final String translationKey; | ||||||
| @@ -55,12 +51,6 @@ public final class TrackingField | |||||||
|         return translationKey; |         return translationKey; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @Deprecated |  | ||||||
|     public String displayName() |  | ||||||
|     { |  | ||||||
|         return StringUtil.translate( translationKey() ); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     private TrackingField( String id, LongFunction<String> format ) |     private TrackingField( String id, LongFunction<String> format ) | ||||||
|     { |     { | ||||||
|         this.id = id; |         this.id = id; | ||||||
| @@ -73,7 +63,7 @@ public final class TrackingField | |||||||
|         return format.apply( value ); |         return format.apply( value ); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     public static TrackingField of( String id, String displayName, LongFunction<String> format ) |     public static TrackingField of( String id, LongFunction<String> format ) | ||||||
|     { |     { | ||||||
|         TrackingField field = new TrackingField( id, format ); |         TrackingField field = new TrackingField( id, format ); | ||||||
|         fields.put( id, field ); |         fields.put( id, field ); | ||||||
|   | |||||||
| @@ -24,7 +24,7 @@ public final class BundledRedstone | |||||||
|  |  | ||||||
|     private BundledRedstone() {} |     private BundledRedstone() {} | ||||||
|  |  | ||||||
|     public static void register( @Nonnull IBundledRedstoneProvider provider ) |     public static synchronized void register( @Nonnull IBundledRedstoneProvider provider ) | ||||||
|     { |     { | ||||||
|         Preconditions.checkNotNull( provider, "provider cannot be null" ); |         Preconditions.checkNotNull( provider, "provider cannot be null" ); | ||||||
|         providers.add( provider ); |         providers.add( provider ); | ||||||
| @@ -32,12 +32,12 @@ public final class BundledRedstone | |||||||
|  |  | ||||||
|     public static int getDefaultOutput( @Nonnull World world, @Nonnull BlockPos pos, @Nonnull EnumFacing side ) |     public static int getDefaultOutput( @Nonnull World world, @Nonnull BlockPos pos, @Nonnull EnumFacing side ) | ||||||
|     { |     { | ||||||
|         return world.isValid( pos ) ? DefaultBundledRedstoneProvider.getDefaultBundledRedstoneOutput( world, pos, side ) : -1; |         return World.isValid( pos ) ? DefaultBundledRedstoneProvider.getDefaultBundledRedstoneOutput( world, pos, side ) : -1; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     private static int getUnmaskedOutput( World world, BlockPos pos, EnumFacing side ) |     private static int getUnmaskedOutput( World world, BlockPos pos, EnumFacing side ) | ||||||
|     { |     { | ||||||
|         if( !world.isValid( pos ) ) return -1; |         if( !World.isValid( pos ) ) return -1; | ||||||
|  |  | ||||||
|         // Try the providers in order: |         // Try the providers in order: | ||||||
|         int combinedSignal = -1; |         int combinedSignal = -1; | ||||||
|   | |||||||
| @@ -12,496 +12,333 @@ 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 net.minecraftforge.common.config.ConfigCategory; | import net.minecraftforge.common.ForgeConfigSpec; | ||||||
| import net.minecraftforge.common.config.ConfigElement; | import net.minecraftforge.eventbus.api.SubscribeEvent; | ||||||
| import net.minecraftforge.common.config.Configuration; | import net.minecraftforge.fml.ModLoadingContext; | ||||||
| import net.minecraftforge.common.config.Property; | import net.minecraftforge.fml.common.Mod; | ||||||
| import net.minecraftforge.fml.client.config.IConfigElement; | import net.minecraftforge.fml.config.ModConfig; | ||||||
|  |  | ||||||
| import java.io.File; | import java.util.Arrays; | ||||||
| import java.util.ArrayList; | import java.util.Collections; | ||||||
| import java.util.List; | import java.util.List; | ||||||
| import java.util.Map; |  | ||||||
| import java.util.Set; |  | ||||||
| import java.util.concurrent.TimeUnit; | import java.util.concurrent.TimeUnit; | ||||||
|  |  | ||||||
| import static dan200.computercraft.ComputerCraft.DEFAULT_HTTP_BLACKLIST; | import static dan200.computercraft.ComputerCraft.DEFAULT_HTTP_BLACKLIST; | ||||||
| import static dan200.computercraft.ComputerCraft.DEFAULT_HTTP_WHITELIST; | import static dan200.computercraft.ComputerCraft.DEFAULT_HTTP_WHITELIST; | ||||||
|  | import static net.minecraftforge.common.ForgeConfigSpec.Builder; | ||||||
|  | import static net.minecraftforge.common.ForgeConfigSpec.ConfigValue; | ||||||
|  |  | ||||||
|  | @Mod.EventBusSubscriber( modid = ComputerCraft.MOD_ID, bus = Mod.EventBusSubscriber.Bus.MOD ) | ||||||
| public final class Config | public final class Config | ||||||
| { | { | ||||||
|     private static final int MODEM_MAX_RANGE = 100000; |     private static final int MODEM_MAX_RANGE = 100000; | ||||||
|  |  | ||||||
|     private static final String CATEGORY_GENERAL = "general"; |     private static final String TRANSLATION_PREFIX = "gui.computercraft.config."; | ||||||
|     private static final String CATEGORY_EXECUTION = "execution"; |  | ||||||
|     private static final String CATEGORY_HTTP = "http"; |  | ||||||
|     private static final String CATEGORY_PERIPHERAL = "peripheral"; |  | ||||||
|     private static final String CATEGORY_TURTLE = "turtle"; |  | ||||||
|  |  | ||||||
|     private static Configuration config; |     private static ConfigValue<Integer> computerSpaceLimit; | ||||||
|  |     private static ConfigValue<Integer> floppySpaceLimit; | ||||||
|  |     private static ConfigValue<Integer> maximumFilesOpen; | ||||||
|  |     private static ConfigValue<Boolean> disableLua51Features; | ||||||
|  |     private static ConfigValue<String> defaultComputerSettings; | ||||||
|  |     private static ConfigValue<Boolean> debugEnabled; | ||||||
|  |     private static ConfigValue<Boolean> logComputerErrors; | ||||||
|  |  | ||||||
|     private static Property computerSpaceLimit; |     private static ConfigValue<Integer> computerThreads; | ||||||
|     private static Property floppySpaceLimit; |     private static ConfigValue<Long> maxMainGlobalTime; | ||||||
|     private static Property maximumFilesOpen; |     private static ConfigValue<Long> maxMainComputerTime; | ||||||
|     private static Property disableLua51Features; |  | ||||||
|     private static Property defaultComputerSettings; |  | ||||||
|     private static Property debugEnabled; |  | ||||||
|     private static Property logComputerErrors; |  | ||||||
|  |  | ||||||
|     private static Property computerThreads; |     private static ConfigValue<Boolean> httpEnabled; | ||||||
|     private static Property maxMainGlobalTime; |     private static ConfigValue<Boolean> httpWebsocketEnabled; | ||||||
|     private static Property maxMainComputerTime; |     private static ConfigValue<List<? extends String>> httpWhitelist; | ||||||
|  |     private static ConfigValue<List<? extends String>> httpBlacklist; | ||||||
|  |  | ||||||
|     private static Property httpEnable; |     private static ConfigValue<Integer> httpTimeout; | ||||||
|     private static Property httpWebsocketEnable; |     private static ConfigValue<Integer> httpMaxRequests; | ||||||
|     private static Property httpWhitelist; |     private static ConfigValue<Long> httpMaxDownload; | ||||||
|     private static Property httpBlacklist; |     private static ConfigValue<Long> httpMaxUpload; | ||||||
|  |     private static ConfigValue<Integer> httpMaxWebsockets; | ||||||
|  |     private static ConfigValue<Integer> httpMaxWebsocketMessage; | ||||||
|  |  | ||||||
|     private static Property httpTimeout; |     private static ConfigValue<Boolean> commandBlockEnabled; | ||||||
|     private static Property httpMaxRequests; |     private static ConfigValue<Integer> modemRange; | ||||||
|     private static Property httpMaxDownload; |     private static ConfigValue<Integer> modemHighAltitudeRange; | ||||||
|     private static Property httpMaxUpload; |     private static ConfigValue<Integer> modemRangeDuringStorm; | ||||||
|     private static Property httpMaxWebsockets; |     private static ConfigValue<Integer> modemHighAltitudeRangeDuringStorm; | ||||||
|     private static Property httpMaxWebsocketMessage; |     private static ConfigValue<Integer> maxNotesPerTick; | ||||||
|  |  | ||||||
|     private static Property commandBlockEnabled; |     private static ConfigValue<Boolean> turtlesNeedFuel; | ||||||
|     private static Property modemRange; |     private static ConfigValue<Integer> turtleFuelLimit; | ||||||
|     private static Property modemHighAltitudeRange; |     private static ConfigValue<Integer> advancedTurtleFuelLimit; | ||||||
|     private static Property modemRangeDuringStorm; |     private static ConfigValue<Boolean> turtlesObeyBlockProtection; | ||||||
|     private static Property modemHighAltitudeRangeDuringStorm; |     private static ConfigValue<Boolean> turtlesCanPush; | ||||||
|     private static Property maxNotesPerTick; |     private static ConfigValue<List<? extends String>> turtleDisabledActions; | ||||||
|  |  | ||||||
|     private static Property turtlesNeedFuel; |     private static final ForgeConfigSpec spec; | ||||||
|     private static Property turtleFuelLimit; |  | ||||||
|     private static Property advancedTurtleFuelLimit; |  | ||||||
|     private static Property turtlesObeyBlockProtection; |  | ||||||
|     private static Property turtlesCanPush; |  | ||||||
|     private static Property turtleDisabledActions; |  | ||||||
|  |  | ||||||
|     private Config() {} |     private Config() {} | ||||||
|  |  | ||||||
|     public static void load( File configFile ) |     static | ||||||
|     { |     { | ||||||
|         config = new Configuration( configFile, ComputerCraft.getVersion() ); |         Builder builder = new Builder(); | ||||||
|  |  | ||||||
|         config.load(); |  | ||||||
|  |  | ||||||
|         { // General computers |         { // General computers | ||||||
|             renameProperty( CATEGORY_GENERAL, "computerSpaceLimit", CATEGORY_GENERAL, "computer_space_limit" ); |             computerSpaceLimit = builder | ||||||
|             renameProperty( CATEGORY_GENERAL, "floppySpaceLimit", CATEGORY_GENERAL, "floppy_space_limit" ); |                 .comment( "The disk space limit for computers and turtles, in bytes" ) | ||||||
|             renameProperty( CATEGORY_GENERAL, "maximumFilesOpen", CATEGORY_GENERAL, "maximum_open_files" ); |                 .translation( TRANSLATION_PREFIX + "computer_space_limit" ) | ||||||
|             renameProperty( CATEGORY_GENERAL, "debug_enable", CATEGORY_GENERAL, "debug_enabled" ); |                 .define( "computer_space_limit", ComputerCraft.computerSpaceLimit ); | ||||||
|             renameProperty( CATEGORY_GENERAL, "logPeripheralErrors", CATEGORY_GENERAL, "log_computer_errors" ); |  | ||||||
|  |  | ||||||
|             computerSpaceLimit = config.get( CATEGORY_GENERAL, "computer_space_limit", ComputerCraft.computerSpaceLimit ); |             floppySpaceLimit = builder | ||||||
|             computerSpaceLimit.setComment( "The disk space limit for computers and turtles, in bytes" ); |                 .comment( "The disk space limit for floppy disks, in bytes" ) | ||||||
|  |                 .translation( TRANSLATION_PREFIX + "floppy_space_limit" ) | ||||||
|  |                 .define( "floppy_space_limit", ComputerCraft.floppySpaceLimit ); | ||||||
|  |  | ||||||
|             floppySpaceLimit = config.get( CATEGORY_GENERAL, "floppy_space_limit", ComputerCraft.floppySpaceLimit ); |             maximumFilesOpen = builder | ||||||
|             floppySpaceLimit.setComment( "The disk space limit for floppy disks, in bytes" ); |                 .comment( "Set how many files a computer can have open at the same time. Set to 0 for unlimited." ) | ||||||
|  |                 .translation( TRANSLATION_PREFIX + "maximum_open_files" ) | ||||||
|  |                 .defineInRange( "maximum_open_files", ComputerCraft.maximumFilesOpen, 0, Integer.MAX_VALUE ); | ||||||
|  |  | ||||||
|             maximumFilesOpen = config.get( CATEGORY_GENERAL, "maximum_open_files", ComputerCraft.maximumFilesOpen ); |             disableLua51Features = builder | ||||||
|             maximumFilesOpen.setComment( "Set how many files a computer can have open at the same time. Set to 0 for unlimited." ); |                 .comment( "Set this to true to disable Lua 5.1 functions that will be removed in a future update. " + | ||||||
|             maximumFilesOpen.setMinValue( 0 ); |                     "Useful for ensuring forward compatibility of your programs now." ) | ||||||
|  |                 .define( "disable_lua51_features", ComputerCraft.disable_lua51_features ); | ||||||
|  |  | ||||||
|             disableLua51Features = config.get( CATEGORY_GENERAL, "disable_lua51_features", ComputerCraft.disable_lua51_features ); |             defaultComputerSettings = builder | ||||||
|             disableLua51Features.setComment( "Set this to true to disable Lua 5.1 functions that will be removed in a future " + |                 .comment( "A comma separated list of default system settings to set on new computers. Example: " + | ||||||
|                 "update. Useful for ensuring forward compatibility of your programs now." ); |                     "\"shell.autocomplete=false,lua.autocomplete=false,edit.autocomplete=false\" will disable all " + | ||||||
|  |                     "autocompletion" ) | ||||||
|  |                 .define( "default_computer_settings", ComputerCraft.default_computer_settings ); | ||||||
|  |  | ||||||
|             defaultComputerSettings = config.get( CATEGORY_GENERAL, "default_computer_settings", ComputerCraft.default_computer_settings ); |             debugEnabled = builder | ||||||
|             defaultComputerSettings.setComment( "A comma separated list of default system settings to set on new computers. Example: " + |                 .comment( "Enable Lua's debug library. This is sandboxed to each computer, so is generally safe to be used by players." ) | ||||||
|                 "\"shell.autocomplete=false,lua.autocomplete=false,edit.autocomplete=false\" will disable all autocompletion" ); |                 .define( "debug_enabled", ComputerCraft.debug_enable ); | ||||||
|  |  | ||||||
|             debugEnabled = config.get( CATEGORY_GENERAL, "debug_enabled", ComputerCraft.debug_enable ); |             logComputerErrors = builder | ||||||
|             debugEnabled.setComment( "Enable Lua's debug library. This is sandboxed to each computer, so is generally safe to be used by players." ); |                 .comment( "Log exceptions thrown by peripherals and other Lua objects.\n" + | ||||||
|  |                     "This makes it easier for mod authors to debug problems, but may result in log spam should people use buggy methods." ) | ||||||
|             logComputerErrors = config.get( CATEGORY_GENERAL, "log_computer_errors", ComputerCraft.logPeripheralErrors ); |                 .define( "log_computer_errors", ComputerCraft.logPeripheralErrors ); | ||||||
|             logComputerErrors.setComment( "Log exceptions thrown by peripherals and other Lua objects.\n" + |  | ||||||
|                 "This makes it easier for mod authors to debug problems, but may result in log spam should people use buggy methods." ); |  | ||||||
|  |  | ||||||
|             setOrder( |  | ||||||
|                 CATEGORY_GENERAL, |  | ||||||
|                 computerSpaceLimit, floppySpaceLimit, maximumFilesOpen, |  | ||||||
|                 disableLua51Features, defaultComputerSettings, debugEnabled, logComputerErrors |  | ||||||
|             ); |  | ||||||
|         } |         } | ||||||
|  |  | ||||||
|         { // Execution |         { | ||||||
|             renameProperty( CATEGORY_GENERAL, "computer_threads", CATEGORY_EXECUTION, "computer_threads" ); |             builder.comment( "Controls execution behaviour of computers. This is largely intended for fine-tuning " + | ||||||
|  |                 "servers, and generally shouldn't need to be touched" ); | ||||||
|  |             builder.push( "execution" ); | ||||||
|  |  | ||||||
|             config.getCategory( CATEGORY_EXECUTION ) |             computerThreads = builder | ||||||
|                 .setComment( "Controls execution behaviour of computers. This is largely intended for fine-tuning " + |                 .comment( "Set the number of threads computers can run on. A higher number means more computers can run " + | ||||||
|                     "servers, and generally shouldn't need to be touched" ); |                     "at once, but may induce lag.\n" + | ||||||
|  |                     "Please note that some mods may not work with a thread count higher than 1. Use with caution." ) | ||||||
|  |                 .worldRestart() | ||||||
|  |                 .defineInRange( "computer_threads", ComputerCraft.computer_threads, 1, Integer.MAX_VALUE ); | ||||||
|  |  | ||||||
|             computerThreads = config.get( CATEGORY_EXECUTION, "computer_threads", ComputerCraft.computer_threads ); |             maxMainGlobalTime = builder | ||||||
|             computerThreads |                 .comment( "The maximum time that can be spent executing tasks in a single tick, in milliseconds.\n" + | ||||||
|                 .setMinValue( 1 ) |                     "Note, we will quite possibly go over this limit, as there's no way to tell how long a will take " + | ||||||
|                 .setRequiresMcRestart( true ) |                     "- this aims to be the upper bound of the average time." ) | ||||||
|                 .setComment( "Set the number of threads computers can run on. A higher number means more computers can " + |                 .defineInRange( "max_main_global_time", TimeUnit.NANOSECONDS.toMillis( ComputerCraft.maxMainGlobalTime ), 1, Long.MAX_VALUE ); | ||||||
|                     "run at once, but may induce lag.\n" + |  | ||||||
|                     "Please note that some mods may not work with a thread count higher than 1. Use with caution." ); |  | ||||||
|  |  | ||||||
|             maxMainGlobalTime = config.get( CATEGORY_EXECUTION, "max_main_global_time", (int) TimeUnit.NANOSECONDS.toMillis( ComputerCraft.maxMainGlobalTime ) ); |             maxMainComputerTime = builder | ||||||
|             maxMainGlobalTime |                 .comment( "The ideal maximum time a computer can execute for in a tick, in milliseconds.\n" + | ||||||
|                 .setMinValue( 1 ) |                     "Note, we will quite possibly go over this limit, as there's no way to tell how long a will take " + | ||||||
|                 .setComment( "The maximum time that can be spent executing tasks in a single tick, in milliseconds.\n" + |                     "- this aims to be the upper bound of the average time." ) | ||||||
|                     "Note, we will quite possibly go over this limit, as there's no way to tell how long a will take - this aims " + |                 .defineInRange( "max_main_computer_time", TimeUnit.NANOSECONDS.toMillis( ComputerCraft.maxMainComputerTime ), 1, Long.MAX_VALUE ); | ||||||
|                     "to be the upper bound of the average time." ); |  | ||||||
|  |  | ||||||
|             maxMainComputerTime = config.get( CATEGORY_EXECUTION, "max_main_computer_time", (int) TimeUnit.NANOSECONDS.toMillis( ComputerCraft.maxMainComputerTime ) ); |             builder.pop(); | ||||||
|             maxMainComputerTime |  | ||||||
|                 .setMinValue( 1 ) |  | ||||||
|                 .setComment( "The ideal maximum time a computer can execute for in a tick, in milliseconds.\n" + |  | ||||||
|                     "Note, we will quite possibly go over this limit, as there's no way to tell how long a will take - this aims " + |  | ||||||
|                     "to be the upper bound of the average time." ); |  | ||||||
|  |  | ||||||
|             setOrder( |  | ||||||
|                 CATEGORY_EXECUTION, |  | ||||||
|                 computerThreads, maxMainGlobalTime, maxMainComputerTime |  | ||||||
|             ); |  | ||||||
|         } |         } | ||||||
|  |  | ||||||
|         { // HTTP |         { // HTTP | ||||||
|             renameProperty( CATEGORY_GENERAL, "http_enable", CATEGORY_HTTP, "enabled" ); |             builder.comment( "Controls the HTTP API" ); | ||||||
|             renameProperty( CATEGORY_GENERAL, "http_websocket_enable", CATEGORY_HTTP, "websocket_enabled" ); |             builder.push( "http" ); | ||||||
|             renameProperty( CATEGORY_GENERAL, "http_whitelist", CATEGORY_HTTP, "whitelist" ); |  | ||||||
|             renameProperty( CATEGORY_GENERAL, "http_blacklist", CATEGORY_HTTP, "blacklist" ); |  | ||||||
|  |  | ||||||
|             config.getCategory( CATEGORY_HTTP ) |             httpEnabled = builder | ||||||
|                 .setComment( "Controls the HTTP API" ); |                 .comment( "Enable the \"http\" API on Computers (see \"http_whitelist\" and \"http_blacklist\" for more " + | ||||||
|  |                     "fine grained control than this)" ) | ||||||
|  |                 .define( "enabled", ComputerCraft.http_enable ); | ||||||
|  |  | ||||||
|             httpEnable = config.get( CATEGORY_HTTP, "enabled", ComputerCraft.http_enable ); |             httpWebsocketEnabled = builder | ||||||
|             httpEnable.setComment( "Enable the \"http\" API on Computers (see \"http_whitelist\" and \"http_blacklist\" for " + |                 .comment( "Enable use of http websockets. This requires the \"http_enable\" option to also be true." ) | ||||||
|                 "more fine grained control than this)" ); |                 .define( "websocket_enabled", ComputerCraft.http_websocket_enable ); | ||||||
|  |  | ||||||
|             httpWebsocketEnable = config.get( CATEGORY_HTTP, "websocket_enabled", ComputerCraft.http_websocket_enable ); |             httpWhitelist = builder | ||||||
|             httpWebsocketEnable.setComment( "Enable use of http websockets. This requires the \"http_enable\" option to also be true." ); |                 .comment( "A list of wildcards for domains or IP ranges that can be accessed through the \"http\" API on Computers.\n" + | ||||||
|  |                     "Set this to \"*\" to access to the entire internet. Example: \"*.pastebin.com\" will restrict access to just subdomains of pastebin.com.\n" + | ||||||
|  |                     "You can use domain names (\"pastebin.com\"), wilcards (\"*.pastebin.com\") or CIDR notation (\"127.0.0.0/8\")." ) | ||||||
|  |                 .defineList( "whitelist", Arrays.asList( DEFAULT_HTTP_WHITELIST ), x -> true ); | ||||||
|  |  | ||||||
|             httpWhitelist = config.get( CATEGORY_HTTP, "whitelist", DEFAULT_HTTP_WHITELIST ); |             httpBlacklist = builder | ||||||
|             httpWhitelist.setComment( "A list of wildcards for domains or IP ranges that can be accessed through the " + |                 .comment( "A list of wildcards for domains or IP ranges that cannot be accessed through the \"http\" API on Computers.\n" + | ||||||
|                 "\"http\" API on Computers.\n" + |                     "If this is empty then all whitelisted domains will be accessible. Example: \"*.github.com\" will block access to all subdomains of github.com.\n" + | ||||||
|                 "Set this to \"*\" to access to the entire internet. Example: \"*.pastebin.com\" will restrict access to " + |                     "You can use domain names (\"pastebin.com\"), wilcards (\"*.pastebin.com\") or CIDR notation (\"127.0.0.0/8\")." ) | ||||||
|                 "just subdomains of pastebin.com.\n" + |                 .defineList( "blacklist", Arrays.asList( DEFAULT_HTTP_BLACKLIST ), x -> true ); | ||||||
|                 "You can use domain names (\"pastebin.com\"), wilcards (\"*.pastebin.com\") or CIDR notation (\"127.0.0.0/8\")." ); |  | ||||||
|  |  | ||||||
|             httpBlacklist = config.get( CATEGORY_HTTP, "blacklist", DEFAULT_HTTP_BLACKLIST ); |             httpTimeout = builder | ||||||
|             httpBlacklist.setComment( "A list of wildcards for domains or IP ranges that cannot be accessed through the " + |                 .comment( "The period of time (in milliseconds) to wait before a HTTP request times out. Set to 0 for unlimited." ) | ||||||
|                 "\"http\" API on Computers.\n" + |                 .defineInRange( "timeout", ComputerCraft.httpTimeout, 0, Integer.MAX_VALUE ); | ||||||
|                 "If this is empty then all whitelisted domains will be accessible. Example: \"*.github.com\" will block " + |  | ||||||
|                 "access to all subdomains of github.com.\n" + |  | ||||||
|                 "You can use domain names (\"pastebin.com\"), wilcards (\"*.pastebin.com\") or CIDR notation (\"127.0.0.0/8\")." ); |  | ||||||
|  |  | ||||||
|             httpTimeout = config.get( CATEGORY_HTTP, "timeout", ComputerCraft.httpTimeout ); |             httpMaxRequests = builder | ||||||
|             httpTimeout.setComment( "The period of time (in milliseconds) to wait before a HTTP request times out. Set to 0 for unlimited." ); |                 .comment( "The number of http requests a computer can make at one time. Additional requests will be queued, and sent when the running requests have finished. Set to 0 for unlimited." ) | ||||||
|             httpTimeout.setMinValue( 0 ); |                 .defineInRange( "max_requests", ComputerCraft.httpMaxRequests, 0, Integer.MAX_VALUE ); | ||||||
|  |  | ||||||
|             httpMaxRequests = config.get( CATEGORY_HTTP, "max_requests", ComputerCraft.httpMaxRequests ); |             httpMaxDownload = builder | ||||||
|             httpMaxRequests.setComment( "The number of http requests a computer can make at one time. Additional requests " + |                 .comment( "The maximum size (in bytes) that a computer can download in a single request. Note that responses may receive more data than allowed, but this data will not be returned to the client." ) | ||||||
|                 "will be queued, and sent when the running requests have finished. Set to 0 for unlimited." ); |                 .defineInRange( "max_download", ComputerCraft.httpMaxDownload, 0, Long.MAX_VALUE ); | ||||||
|             httpMaxRequests.setMinValue( 0 ); |  | ||||||
|  |  | ||||||
|             httpMaxDownload = config.get( CATEGORY_HTTP, "max_download", (int) ComputerCraft.httpMaxDownload ); |             httpMaxUpload = builder | ||||||
|             httpMaxDownload.setComment( "The maximum size (in bytes) that a computer can download in a single request. " + |                 .comment( "The maximum size (in bytes) that a computer can upload in a single request. This includes headers and POST text." ) | ||||||
|                 "Note that responses may receive more data than allowed, but this data will not be returned to the client." ); |                 .defineInRange( "max_upload", ComputerCraft.httpMaxUpload, 0, Long.MAX_VALUE ); | ||||||
|             httpMaxDownload.setMinValue( 0 ); |  | ||||||
|  |  | ||||||
|             httpMaxUpload = config.get( CATEGORY_HTTP, "max_upload", (int) ComputerCraft.httpMaxUpload ); |             httpMaxWebsockets = builder | ||||||
|             httpMaxUpload.setComment( "The maximum size (in bytes) that a computer can upload in a single request. This " + |                 .comment( "The number of websockets a computer can have open at one time. Set to 0 for unlimited." ) | ||||||
|                 "includes headers and POST text." ); |                 .defineInRange( "max_websockets", ComputerCraft.httpMaxWebsockets, 1, Integer.MAX_VALUE ); | ||||||
|             httpMaxUpload.setMinValue( 0 ); |  | ||||||
|  |  | ||||||
|             httpMaxWebsockets = config.get( CATEGORY_HTTP, "max_websockets", ComputerCraft.httpMaxWebsockets ); |             httpMaxWebsocketMessage = builder | ||||||
|             httpMaxWebsockets.setComment( "The number of websockets a computer can have open at one time. Set to 0 for unlimited." ); |                 .comment( "The maximum size (in bytes) that a computer can send or receive in one websocket packet." ) | ||||||
|             httpMaxWebsockets.setMinValue( 1 ); |                 .defineInRange( "max_websocket_message", ComputerCraft.httpMaxWebsocketMessage, 0, Websocket.MAX_MESSAGE_SIZE ); | ||||||
|  |  | ||||||
|             httpMaxWebsocketMessage = config.get( CATEGORY_HTTP, "max_websocket_message", ComputerCraft.httpMaxWebsocketMessage ); |             builder.pop(); | ||||||
|             httpMaxWebsocketMessage.setComment( "The maximum size (in bytes) that a computer can send or receive in one websocket packet." ); |  | ||||||
|             httpMaxWebsocketMessage.setMinValue( 0 ); |  | ||||||
|             httpMaxWebsocketMessage.setMaxValue( Websocket.MAX_MESSAGE_SIZE ); |  | ||||||
|  |  | ||||||
|             setOrder( |  | ||||||
|                 CATEGORY_HTTP, |  | ||||||
|                 httpEnable, httpWebsocketEnable, httpWhitelist, httpBlacklist, |  | ||||||
|                 httpTimeout, httpMaxRequests, httpMaxDownload, httpMaxUpload, httpMaxWebsockets, httpMaxWebsocketMessage |  | ||||||
|             ); |  | ||||||
|         } |         } | ||||||
|  |  | ||||||
|         { // Peripherals |         { // Peripherals | ||||||
|             renameProperty( CATEGORY_GENERAL, "enableCommandBlock", CATEGORY_PERIPHERAL, "command_block_enabled" ); |             builder.comment( "Various options relating to peripherals." ); | ||||||
|             renameProperty( CATEGORY_GENERAL, "modem_range", CATEGORY_PERIPHERAL, "modem_range" ); |             builder.push( "peripheral" ); | ||||||
|             renameProperty( CATEGORY_GENERAL, "modem_highAltitudeRange", CATEGORY_PERIPHERAL, "modem_high_altitude_range" ); |  | ||||||
|             renameProperty( CATEGORY_GENERAL, "modem_rangeDuringStorm", CATEGORY_PERIPHERAL, "modem_range_during_storm" ); |  | ||||||
|             renameProperty( CATEGORY_GENERAL, "modem_highAltitudeRangeDuringStorm", CATEGORY_PERIPHERAL, "modem_high_altitude_range_during_storm" ); |  | ||||||
|             renameProperty( CATEGORY_GENERAL, "maxNotesPerTick", CATEGORY_PERIPHERAL, "max_notes_per_tick" ); |  | ||||||
|  |  | ||||||
|             config.getCategory( CATEGORY_PERIPHERAL ) |             commandBlockEnabled = builder | ||||||
|                 .setComment( "Various options relating to peripherals." ); |                 .comment( "Enable Command Block peripheral support" ) | ||||||
|  |                 .define( "command_block_enabled", ComputerCraft.enableCommandBlock ); | ||||||
|  |  | ||||||
|             commandBlockEnabled = config.get( CATEGORY_PERIPHERAL, "command_block_enabled", ComputerCraft.enableCommandBlock ); |             modemRange = builder | ||||||
|             commandBlockEnabled.setComment( "Enable Command Block peripheral support" ); |                 .comment( "The range of Wireless Modems at low altitude in clear weather, in meters" ) | ||||||
|  |                 .defineInRange( "modem_range", ComputerCraft.modem_range, 0, MODEM_MAX_RANGE ); | ||||||
|  |  | ||||||
|             modemRange = config.get( CATEGORY_PERIPHERAL, "modem_range", ComputerCraft.modem_range ); |             modemHighAltitudeRange = builder | ||||||
|             modemRange.setComment( "The range of Wireless Modems at low altitude in clear weather, in meters" ); |                 .comment( "The range of Wireless Modems at maximum altitude in clear weather, in meters" ) | ||||||
|             modemRange.setMinValue( 0 ); |                 .defineInRange( "modem_high_altitude_range", ComputerCraft.modem_highAltitudeRange, 0, MODEM_MAX_RANGE ); | ||||||
|             modemRange.setMaxValue( MODEM_MAX_RANGE ); |  | ||||||
|  |  | ||||||
|             modemHighAltitudeRange = config.get( CATEGORY_PERIPHERAL, "modem_high_altitude_range", ComputerCraft.modem_highAltitudeRange ); |             modemRangeDuringStorm = builder | ||||||
|             modemHighAltitudeRange.setComment( "The range of Wireless Modems at maximum altitude in clear weather, in meters" ); |                 .comment( "The range of Wireless Modems at low altitude in stormy weather, in meters" ) | ||||||
|             modemHighAltitudeRange.setMinValue( 0 ); |                 .defineInRange( "modem_range_during_storm", ComputerCraft.modem_rangeDuringStorm, 0, MODEM_MAX_RANGE ); | ||||||
|             modemHighAltitudeRange.setMaxValue( MODEM_MAX_RANGE ); |  | ||||||
|  |  | ||||||
|             modemRangeDuringStorm = config.get( CATEGORY_PERIPHERAL, "modem_range_during_storm", ComputerCraft.modem_rangeDuringStorm ); |             modemHighAltitudeRangeDuringStorm = builder | ||||||
|             modemRangeDuringStorm.setComment( "The range of Wireless Modems at low altitude in stormy weather, in meters" ); |                 .comment( "The range of Wireless Modems at maximum altitude in stormy weather, in meters" ) | ||||||
|             modemRangeDuringStorm.setMinValue( 0 ); |                 .defineInRange( "modem_high_altitude_range_during_storm", ComputerCraft.modem_highAltitudeRangeDuringStorm, 0, MODEM_MAX_RANGE ); | ||||||
|             modemRangeDuringStorm.setMaxValue( MODEM_MAX_RANGE ); |  | ||||||
|  |  | ||||||
|             modemHighAltitudeRangeDuringStorm = config.get( CATEGORY_PERIPHERAL, "modem_high_altitude_range_during_storm", ComputerCraft.modem_highAltitudeRangeDuringStorm ); |             maxNotesPerTick = builder | ||||||
|             modemHighAltitudeRangeDuringStorm.setComment( "The range of Wireless Modems at maximum altitude in stormy weather, in meters" ); |                 .comment( "Maximum amount of notes a speaker can play at once" ) | ||||||
|             modemHighAltitudeRangeDuringStorm.setMinValue( 0 ); |                 .defineInRange( "max_notes_per_tick", ComputerCraft.maxNotesPerTick, 1, Integer.MAX_VALUE ); | ||||||
|             modemHighAltitudeRangeDuringStorm.setMaxValue( MODEM_MAX_RANGE ); |  | ||||||
|  |  | ||||||
|             maxNotesPerTick = config.get( CATEGORY_PERIPHERAL, "max_notes_per_tick", ComputerCraft.maxNotesPerTick ); |             builder.pop(); | ||||||
|             maxNotesPerTick.setComment( "Maximum amount of notes a speaker can play at once" ); |  | ||||||
|             maxNotesPerTick.setMinValue( 1 ); |  | ||||||
|  |  | ||||||
|             setOrder( |  | ||||||
|                 CATEGORY_PERIPHERAL, |  | ||||||
|                 commandBlockEnabled, modemRange, modemHighAltitudeRange, modemRangeDuringStorm, modemHighAltitudeRangeDuringStorm, maxNotesPerTick |  | ||||||
|             ); |  | ||||||
|         } |         } | ||||||
|  |  | ||||||
|         { // Turtles |         { // Turtles | ||||||
|             renameProperty( CATEGORY_GENERAL, "turtlesNeedFuel", CATEGORY_TURTLE, "need_fuel" ); |             builder.comment( "Various options relating to turtles." ); | ||||||
|             renameProperty( CATEGORY_GENERAL, "turtleFuelLimit", CATEGORY_TURTLE, "normal_fuel_limit" ); |             builder.push( "turtle" ); | ||||||
|             renameProperty( CATEGORY_GENERAL, "advancedTurtleFuelLimit", CATEGORY_TURTLE, "advanced_fuel_limit" ); |  | ||||||
|             renameProperty( CATEGORY_GENERAL, "turtlesObeyBlockProtection", CATEGORY_TURTLE, "obey_block_protection" ); |  | ||||||
|             renameProperty( CATEGORY_GENERAL, "turtlesCanPush", CATEGORY_TURTLE, "can_push" ); |  | ||||||
|             renameProperty( CATEGORY_GENERAL, "turtle_disabled_actions", CATEGORY_TURTLE, "disabled_actions" ); |  | ||||||
|  |  | ||||||
|             config.getCategory( CATEGORY_HTTP ) |             turtlesNeedFuel = builder | ||||||
|                 .setComment( "Various options relating to turtles." ); |                 .comment( "Set whether Turtles require fuel to move" ) | ||||||
|  |                 .define( "need_fuel", ComputerCraft.turtlesNeedFuel ); | ||||||
|  |  | ||||||
|             turtlesNeedFuel = config.get( CATEGORY_TURTLE, "need_fuel", ComputerCraft.turtlesNeedFuel ); |             turtleFuelLimit = builder | ||||||
|             turtlesNeedFuel.setComment( "Set whether Turtles require fuel to move" ); |                 .comment( "The fuel limit for Turtles" ) | ||||||
|  |                 .defineInRange( "normal_fuel_limit", ComputerCraft.turtleFuelLimit, 0, Integer.MAX_VALUE ); | ||||||
|  |  | ||||||
|             turtleFuelLimit = config.get( CATEGORY_TURTLE, "normal_fuel_limit", ComputerCraft.turtleFuelLimit ); |             advancedTurtleFuelLimit = builder | ||||||
|             turtleFuelLimit.setComment( "The fuel limit for Turtles" ); |                 .comment( "The fuel limit for Advanced Turtles" ) | ||||||
|             turtleFuelLimit.setMinValue( 0 ); |                 .defineInRange( "advanced_fuel_limit", ComputerCraft.advancedTurtleFuelLimit, 0, Integer.MAX_VALUE ); | ||||||
|  |  | ||||||
|             advancedTurtleFuelLimit = config.get( CATEGORY_TURTLE, "advanced_fuel_limit", ComputerCraft.advancedTurtleFuelLimit ); |             turtlesObeyBlockProtection = builder | ||||||
|             advancedTurtleFuelLimit.setComment( "The fuel limit for Advanced Turtles" ); |                 .comment( "If set to true, Turtles will be unable to build, dig, or enter protected areas (such as near the server spawn point)" ) | ||||||
|             advancedTurtleFuelLimit.setMinValue( 0 ); |                 .define( "obey_block_protection", ComputerCraft.turtlesObeyBlockProtection ); | ||||||
|  |  | ||||||
|             turtlesObeyBlockProtection = config.get( CATEGORY_TURTLE, "obey_block_protection", ComputerCraft.turtlesObeyBlockProtection ); |             turtlesCanPush = builder | ||||||
|             turtlesObeyBlockProtection.setComment( "If set to true, Turtles will be unable to build, dig, or enter protected " + |                 .comment( "If set to true, Turtles will push entities out of the way instead of stopping if there is space to do so" ) | ||||||
|                 "areas (such as near the server spawn point)" ); |                 .define( "can_push", ComputerCraft.turtlesCanPush ); | ||||||
|  |  | ||||||
|             turtlesCanPush = config.get( CATEGORY_TURTLE, "can_push", ComputerCraft.turtlesCanPush ); |             turtleDisabledActions = builder | ||||||
|             turtlesCanPush.setComment( "If set to true, Turtles will push entities out of the way instead of stopping if " + |                 .comment( "A list of turtle actions which are disabled." ) | ||||||
|                 "there is space to do so" ); |                 .defineList( "disabled_actions", Collections.emptyList(), x -> x instanceof String && getAction( (String) x ) != null ); | ||||||
|  |  | ||||||
|             turtleDisabledActions = config.get( CATEGORY_TURTLE, "disabled_actions", new String[0] ); |             builder.pop(); | ||||||
|             turtleDisabledActions.setComment( "A list of turtle actions which are disabled." ); |  | ||||||
|  |  | ||||||
|             setOrder( |  | ||||||
|                 CATEGORY_TURTLE, |  | ||||||
|                 turtlesNeedFuel, turtleFuelLimit, advancedTurtleFuelLimit, turtlesObeyBlockProtection, turtlesCanPush, turtleDisabledActions |  | ||||||
|             ); |  | ||||||
|         } |         } | ||||||
|  |  | ||||||
|         for( String child : config.getCategoryNames() ) |         spec = builder.build(); | ||||||
|         { |  | ||||||
|             setupLanguage( |  | ||||||
|                 config.getCategory( child ), |  | ||||||
|                 child.equals( CATEGORY_GENERAL ) ? "gui.computercraft:config" : "gui.computercraft:config." + child |  | ||||||
|             ); |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         sync(); |  | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     private static void setOrder( String category, Property... properties ) |     public static void load() | ||||||
|     { |     { | ||||||
|         List<String> names = new ArrayList<>( properties.length ); |         ModLoadingContext.get().registerConfig( ModConfig.Type.COMMON, spec ); | ||||||
|         for( Property property : properties ) names.add( property.getName() ); |  | ||||||
|         config.getCategory( category ).setPropertyOrder( names ); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     private static void renameProperty( String oldCat, String oldProp, String newCat, String newProp ) |  | ||||||
|     { |  | ||||||
|         if( !config.hasCategory( oldCat ) ) return; |  | ||||||
|  |  | ||||||
|         ConfigCategory cat = config.getCategory( oldCat ); |  | ||||||
|         if( !cat.containsKey( oldProp ) ) return; |  | ||||||
|  |  | ||||||
|         Property prop = cat.remove( oldProp ); |  | ||||||
|         prop.setName( newProp ); |  | ||||||
|         config.getCategory( newCat ).put( newProp, prop ); |  | ||||||
|  |  | ||||||
|         // Clean up old categories |  | ||||||
|         if( cat.isEmpty() ) config.removeCategory( cat ); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     private static void setupLanguage( ConfigCategory category, String key ) |  | ||||||
|     { |  | ||||||
|         category.setLanguageKey( key ); |  | ||||||
|         for( Property property : category.getOrderedValues() ) |  | ||||||
|         { |  | ||||||
|             property.setLanguageKey( key + "." + property.getName() ); |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         for( ConfigCategory child : category.getChildren() ) |  | ||||||
|         { |  | ||||||
|             setupLanguage( child, key + "." + child.getName() ); |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     public static void reload() |  | ||||||
|     { |  | ||||||
|         Configuration newConfig = new Configuration( config.getConfigFile(), ComputerCraft.getVersion() ); |  | ||||||
|         Set<String> oldCategories = config.getCategoryNames(), newCategories = newConfig.getCategoryNames(); |  | ||||||
|  |  | ||||||
|         // Sync any categories on the original config |  | ||||||
|         for( String category : oldCategories ) |  | ||||||
|         { |  | ||||||
|             if( newCategories.contains( category ) ) |  | ||||||
|             { |  | ||||||
|                 reloadCategory( config.getCategory( category ), newConfig.getCategory( category ) ); |  | ||||||
|             } |  | ||||||
|             else |  | ||||||
|             { |  | ||||||
|                 for( Property property : config.getCategory( category ).getValues().values() ) property.setToDefault(); |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         // And drop any unexpected ones. |  | ||||||
|         for( String category : newCategories ) |  | ||||||
|         { |  | ||||||
|             if( !oldCategories.contains( category ) ) |  | ||||||
|             { |  | ||||||
|                 ComputerCraft.log.warn( "Cannot sync unknown config category {}", category ); |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         sync(); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     private static void reloadCategory( ConfigCategory oldCat, ConfigCategory newCat ) |  | ||||||
|     { |  | ||||||
|         // Copy the config values across to the original config. |  | ||||||
|         for( Map.Entry<String, Property> child : newCat.getValues().entrySet() ) |  | ||||||
|         { |  | ||||||
|             Property oldProperty = oldCat.get( child.getKey() ), newProperty = child.getValue(); |  | ||||||
|             if( oldProperty.getType() != newProperty.getType() || oldProperty.isList() != newProperty.isList() ) |  | ||||||
|             { |  | ||||||
|                 ComputerCraft.log.warn( |  | ||||||
|                     "Cannot sync config property {} (type changed from {} to {})", |  | ||||||
|                     child.getKey(), getType( oldProperty ), getType( newProperty ) |  | ||||||
|                 ); |  | ||||||
|                 continue; |  | ||||||
|             } |  | ||||||
|  |  | ||||||
|             if( oldProperty.isList() ) |  | ||||||
|             { |  | ||||||
|                 oldProperty.setValues( oldProperty.getStringList() ); |  | ||||||
|             } |  | ||||||
|             else |  | ||||||
|             { |  | ||||||
|                 oldProperty.setValue( newProperty.getString() ); |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         // Reset any missing properties. |  | ||||||
|         for( Map.Entry<String, Property> child : oldCat.getValues().entrySet() ) |  | ||||||
|         { |  | ||||||
|             if( !newCat.containsKey( child.getKey() ) ) child.getValue().setToDefault(); |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     private static String getType( Property property ) |  | ||||||
|     { |  | ||||||
|         return property.getType() + (property.isList() ? " list" : ""); |  | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     public static void sync() |     public static void sync() | ||||||
|     { |     { | ||||||
|         // General |         // General | ||||||
|         ComputerCraft.computerSpaceLimit = computerSpaceLimit.getInt(); |         ComputerCraft.computerSpaceLimit = computerSpaceLimit.get(); | ||||||
|         ComputerCraft.floppySpaceLimit = floppySpaceLimit.getInt(); |         ComputerCraft.floppySpaceLimit = floppySpaceLimit.get(); | ||||||
|         ComputerCraft.maximumFilesOpen = Math.max( 0, maximumFilesOpen.getInt() ); |         ComputerCraft.maximumFilesOpen = maximumFilesOpen.get(); | ||||||
|         ComputerCraft.disable_lua51_features = disableLua51Features.getBoolean(); |         ComputerCraft.disable_lua51_features = disableLua51Features.get(); | ||||||
|         ComputerCraft.default_computer_settings = defaultComputerSettings.getString(); |         ComputerCraft.default_computer_settings = defaultComputerSettings.get(); | ||||||
|         ComputerCraft.debug_enable = debugEnabled.getBoolean(); |         ComputerCraft.debug_enable = debugEnabled.get(); | ||||||
|         ComputerCraft.logPeripheralErrors = logComputerErrors.getBoolean(); |         ComputerCraft.computer_threads = computerThreads.get(); | ||||||
|  |         ComputerCraft.logPeripheralErrors = logComputerErrors.get(); | ||||||
|  |  | ||||||
|         // Execution |         // Execution | ||||||
|         ComputerCraft.computer_threads = computerThreads.getInt(); |         ComputerCraft.computer_threads = computerThreads.get(); | ||||||
|         ComputerCraft.maxMainGlobalTime = TimeUnit.MILLISECONDS.toNanos( Math.max( 1, maxMainGlobalTime.getLong() ) ); |         ComputerCraft.maxMainGlobalTime = TimeUnit.MILLISECONDS.toNanos( maxMainGlobalTime.get() ); | ||||||
|         ComputerCraft.maxMainComputerTime = TimeUnit.MILLISECONDS.toNanos( Math.max( 1, maxMainComputerTime.getLong() ) ); |         ComputerCraft.maxMainComputerTime = TimeUnit.MILLISECONDS.toNanos( maxMainComputerTime.get() ); | ||||||
|  |  | ||||||
|         // HTTP |         // HTTP | ||||||
|         ComputerCraft.http_enable = httpEnable.getBoolean(); |         ComputerCraft.http_enable = httpEnabled.get(); | ||||||
|         ComputerCraft.http_websocket_enable = httpWebsocketEnable.getBoolean(); |         ComputerCraft.http_websocket_enable = httpWebsocketEnabled.get(); | ||||||
|         ComputerCraft.http_whitelist = new AddressPredicate( httpWhitelist.getStringList() ); |         ComputerCraft.http_whitelist = new AddressPredicate( httpWhitelist.get() ); | ||||||
|         ComputerCraft.http_blacklist = new AddressPredicate( httpBlacklist.getStringList() ); |         ComputerCraft.http_blacklist = new AddressPredicate( httpBlacklist.get() ); | ||||||
|  |  | ||||||
|         ComputerCraft.httpTimeout = Math.max( 0, httpTimeout.getInt() ); |         ComputerCraft.httpTimeout = httpTimeout.get(); | ||||||
|         ComputerCraft.httpMaxRequests = Math.max( 1, httpMaxRequests.getInt() ); |         ComputerCraft.httpMaxRequests = httpMaxRequests.get(); | ||||||
|         ComputerCraft.httpMaxDownload = Math.max( 0, httpMaxDownload.getLong() ); |         ComputerCraft.httpMaxDownload = httpMaxDownload.get(); | ||||||
|         ComputerCraft.httpMaxUpload = Math.max( 0, httpMaxUpload.getLong() ); |         ComputerCraft.httpMaxUpload = httpMaxUpload.get(); | ||||||
|         ComputerCraft.httpMaxWebsockets = Math.max( 1, httpMaxWebsockets.getInt() ); |         ComputerCraft.httpMaxWebsockets = httpMaxWebsockets.get(); | ||||||
|         ComputerCraft.httpMaxWebsocketMessage = Math.max( 0, httpMaxWebsocketMessage.getInt() ); |         ComputerCraft.httpMaxWebsocketMessage = httpMaxWebsocketMessage.get(); | ||||||
|  |  | ||||||
|         // Peripheral |         // Peripheral | ||||||
|         ComputerCraft.enableCommandBlock = commandBlockEnabled.getBoolean(); |         ComputerCraft.enableCommandBlock = commandBlockEnabled.get(); | ||||||
|         ComputerCraft.maxNotesPerTick = Math.max( 1, maxNotesPerTick.getInt() ); |         ComputerCraft.maxNotesPerTick = maxNotesPerTick.get(); | ||||||
|         ComputerCraft.modem_range = Math.min( modemRange.getInt(), MODEM_MAX_RANGE ); |         ComputerCraft.modem_range = modemRange.get(); | ||||||
|         ComputerCraft.modem_highAltitudeRange = Math.min( modemHighAltitudeRange.getInt(), MODEM_MAX_RANGE ); |         ComputerCraft.modem_highAltitudeRange = modemHighAltitudeRange.get(); | ||||||
|         ComputerCraft.modem_rangeDuringStorm = Math.min( modemRangeDuringStorm.getInt(), MODEM_MAX_RANGE ); |         ComputerCraft.modem_rangeDuringStorm = modemRangeDuringStorm.get(); | ||||||
|         ComputerCraft.modem_highAltitudeRangeDuringStorm = Math.min( modemHighAltitudeRangeDuringStorm.getInt(), MODEM_MAX_RANGE ); |         ComputerCraft.modem_highAltitudeRangeDuringStorm = modemHighAltitudeRangeDuringStorm.get(); | ||||||
|  |  | ||||||
|         // Turtles |         // Turtles | ||||||
|         ComputerCraft.turtlesNeedFuel = turtlesNeedFuel.getBoolean(); |         ComputerCraft.turtlesNeedFuel = turtlesNeedFuel.get(); | ||||||
|         ComputerCraft.turtleFuelLimit = turtleFuelLimit.getInt(); |         ComputerCraft.turtleFuelLimit = turtleFuelLimit.get(); | ||||||
|         ComputerCraft.advancedTurtleFuelLimit = advancedTurtleFuelLimit.getInt(); |         ComputerCraft.advancedTurtleFuelLimit = advancedTurtleFuelLimit.get(); | ||||||
|         ComputerCraft.turtlesObeyBlockProtection = turtlesObeyBlockProtection.getBoolean(); |         ComputerCraft.turtlesObeyBlockProtection = turtlesObeyBlockProtection.get(); | ||||||
|         ComputerCraft.turtlesCanPush = turtlesCanPush.getBoolean(); |         ComputerCraft.turtlesCanPush = turtlesCanPush.get(); | ||||||
|  |  | ||||||
|         ComputerCraft.turtleDisabledActions.clear(); |         ComputerCraft.turtleDisabledActions.clear(); | ||||||
|         Converter<String, String> converter = CaseFormat.LOWER_CAMEL.converterTo( CaseFormat.UPPER_UNDERSCORE ); |         for( String value : turtleDisabledActions.get() ) ComputerCraft.turtleDisabledActions.add( getAction( value ) ); | ||||||
|         for( String value : turtleDisabledActions.getStringList() ) |  | ||||||
|         { |  | ||||||
|             try |  | ||||||
|             { |  | ||||||
|                 ComputerCraft.turtleDisabledActions.add( TurtleAction.valueOf( converter.convert( value ) ) ); |  | ||||||
|             } |  | ||||||
|             catch( IllegalArgumentException e ) |  | ||||||
|             { |  | ||||||
|                 ComputerCraft.log.error( "Unknown turtle action " + value ); |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         config.save(); |  | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     public static List<IConfigElement> getConfigElements() |     @SubscribeEvent | ||||||
|  |     public static void sync( ModConfig.Loading event ) | ||||||
|     { |     { | ||||||
|         ArrayList<IConfigElement> elements = new ArrayList<>(); |         sync(); | ||||||
|  |  | ||||||
|         // Add all child categories |  | ||||||
|         for( String categoryName : config.getCategoryNames() ) |  | ||||||
|         { |  | ||||||
|             if( categoryName.equals( CATEGORY_GENERAL ) ) continue; |  | ||||||
|             ConfigCategory category = config.getCategory( categoryName ); |  | ||||||
|             elements.add( new ConfigElement( category ) ); |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         // Add the general category |  | ||||||
|         for( Property property : config.getCategory( CATEGORY_GENERAL ).getOrderedValues() ) |  | ||||||
|         { |  | ||||||
|             elements.add( new ConfigElement( property ) ); |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         return elements; |  | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     @SubscribeEvent | ||||||
|  |     public static void sync( ModConfig.ConfigReloading event ) | ||||||
|  |     { | ||||||
|  |         sync(); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     private static final Converter<String, String> converter = CaseFormat.LOWER_CAMEL.converterTo( CaseFormat.UPPER_UNDERSCORE ); | ||||||
|  |  | ||||||
|  |     private static TurtleAction getAction( String value ) | ||||||
|  |     { | ||||||
|  |         try | ||||||
|  |         { | ||||||
|  |             return TurtleAction.valueOf( converter.convert( value ) ); | ||||||
|  |         } | ||||||
|  |         catch( IllegalArgumentException e ) | ||||||
|  |         { | ||||||
|  |             return null; | ||||||
|  |         } | ||||||
|  |     } | ||||||
| } | } | ||||||
|   | |||||||
| @@ -22,7 +22,7 @@ public final class MediaProviders | |||||||
|  |  | ||||||
|     private MediaProviders() {} |     private MediaProviders() {} | ||||||
|  |  | ||||||
|     public static void register( @Nonnull IMediaProvider provider ) |     public static synchronized void register( @Nonnull IMediaProvider provider ) | ||||||
|     { |     { | ||||||
|         Objects.requireNonNull( provider, "provider cannot be null" ); |         Objects.requireNonNull( provider, "provider cannot be null" ); | ||||||
|         providers.add( provider ); |         providers.add( provider ); | ||||||
|   | |||||||
| @@ -15,23 +15,24 @@ import net.minecraft.world.World; | |||||||
|  |  | ||||||
| import javax.annotation.Nonnull; | import javax.annotation.Nonnull; | ||||||
| import java.util.Collection; | import java.util.Collection; | ||||||
|  | import java.util.LinkedHashSet; | ||||||
| import java.util.Objects; | import java.util.Objects; | ||||||
|  |  | ||||||
| public final class Peripherals | public final class Peripherals | ||||||
| { | { | ||||||
|     private static final Collection<IPeripheralProvider> providers = ComputerCraft.peripheralProviders; |     private static final Collection<IPeripheralProvider> providers = new LinkedHashSet<>(); | ||||||
|  |  | ||||||
|     private Peripherals() {} |     private Peripherals() {} | ||||||
|  |  | ||||||
|     public static void register( @Nonnull IPeripheralProvider provider ) |     public static synchronized void register( @Nonnull IPeripheralProvider provider ) | ||||||
|     { |     { | ||||||
|         Objects.requireNonNull( provider, "provider cannot be null" ); |         Objects.requireNonNull( provider, "provider cannot be null" ); | ||||||
|         if( !providers.contains( provider ) ) providers.add( provider ); |         providers.add( provider ); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     public static IPeripheral getPeripheral( World world, BlockPos pos, EnumFacing side ) |     public static IPeripheral getPeripheral( World world, BlockPos pos, EnumFacing side ) | ||||||
|     { |     { | ||||||
|         return world.isValid( pos ) && !world.isRemote ? getPeripheralAt( world, pos, side ) : null; |         return World.isValid( pos ) && !world.isRemote ? getPeripheralAt( world, pos, side ) : null; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     private static IPeripheral getPeripheralAt( World world, BlockPos pos, EnumFacing side ) |     private static IPeripheral getPeripheralAt( World world, BlockPos pos, EnumFacing side ) | ||||||
|   | |||||||
| @@ -10,8 +10,8 @@ import dan200.computercraft.ComputerCraft; | |||||||
| import dan200.computercraft.api.pocket.IPocketUpgrade; | import dan200.computercraft.api.pocket.IPocketUpgrade; | ||||||
| import dan200.computercraft.shared.util.InventoryUtil; | import dan200.computercraft.shared.util.InventoryUtil; | ||||||
| import net.minecraft.item.ItemStack; | import net.minecraft.item.ItemStack; | ||||||
| import net.minecraftforge.fml.common.Loader; | import net.minecraftforge.fml.ModContainer; | ||||||
| import net.minecraftforge.fml.common.ModContainer; | import net.minecraftforge.fml.ModLoadingContext; | ||||||
|  |  | ||||||
| import javax.annotation.Nonnull; | import javax.annotation.Nonnull; | ||||||
| import javax.annotation.Nullable; | import javax.annotation.Nullable; | ||||||
| @@ -24,7 +24,7 @@ public final class PocketUpgrades | |||||||
|  |  | ||||||
|     private PocketUpgrades() {} |     private PocketUpgrades() {} | ||||||
|  |  | ||||||
|     public static void register( @Nonnull IPocketUpgrade upgrade ) |     public static synchronized void register( @Nonnull IPocketUpgrade upgrade ) | ||||||
|     { |     { | ||||||
|         Objects.requireNonNull( upgrade, "upgrade cannot be null" ); |         Objects.requireNonNull( upgrade, "upgrade cannot be null" ); | ||||||
|  |  | ||||||
| @@ -37,7 +37,7 @@ public final class PocketUpgrades | |||||||
|  |  | ||||||
|         upgrades.put( id, upgrade ); |         upgrades.put( id, upgrade ); | ||||||
|  |  | ||||||
|         ModContainer mc = Loader.instance().activeModContainer(); |         ModContainer mc = ModLoadingContext.get().getActiveContainer(); | ||||||
|         if( mc != null && mc.getModId() != null ) upgradeOwners.put( upgrade, mc.getModId() ); |         if( mc != null && mc.getModId() != null ) upgradeOwners.put( upgrade, mc.getModId() ); | ||||||
|     } |     } | ||||||
|  |  | ||||||
| @@ -74,8 +74,8 @@ public final class PocketUpgrades | |||||||
|     public static Iterable<IPocketUpgrade> getVanillaUpgrades() |     public static Iterable<IPocketUpgrade> getVanillaUpgrades() | ||||||
|     { |     { | ||||||
|         List<IPocketUpgrade> vanilla = new ArrayList<>(); |         List<IPocketUpgrade> vanilla = new ArrayList<>(); | ||||||
|         vanilla.add( ComputerCraft.PocketUpgrades.wirelessModem ); |         vanilla.add( ComputerCraft.PocketUpgrades.wirelessModemNormal ); | ||||||
|         vanilla.add( ComputerCraft.PocketUpgrades.advancedModem ); |         vanilla.add( ComputerCraft.PocketUpgrades.wirelessModemAdvanced ); | ||||||
|         vanilla.add( ComputerCraft.PocketUpgrades.speaker ); |         vanilla.add( ComputerCraft.PocketUpgrades.speaker ); | ||||||
|         return vanilla; |         return vanilla; | ||||||
|     } |     } | ||||||
|   | |||||||
| @@ -8,52 +8,53 @@ package dan200.computercraft.shared; | |||||||
|  |  | ||||||
| import dan200.computercraft.ComputerCraft; | import dan200.computercraft.ComputerCraft; | ||||||
| import dan200.computercraft.api.ComputerCraftAPI; | import dan200.computercraft.api.ComputerCraftAPI; | ||||||
| import dan200.computercraft.shared.computer.blocks.BlockCommandComputer; |  | ||||||
| import dan200.computercraft.shared.computer.blocks.BlockComputer; | import dan200.computercraft.shared.computer.blocks.BlockComputer; | ||||||
| import dan200.computercraft.shared.computer.blocks.TileCommandComputer; | import dan200.computercraft.shared.computer.blocks.TileCommandComputer; | ||||||
| import dan200.computercraft.shared.computer.blocks.TileComputer; | import dan200.computercraft.shared.computer.blocks.TileComputer; | ||||||
| import dan200.computercraft.shared.computer.items.ItemCommandComputer; | import dan200.computercraft.shared.computer.core.ComputerFamily; | ||||||
| import dan200.computercraft.shared.computer.items.ItemComputer; | import dan200.computercraft.shared.computer.items.ItemComputer; | ||||||
| import dan200.computercraft.shared.media.items.ItemDiskExpanded; | import dan200.computercraft.shared.media.items.ItemDisk; | ||||||
| import dan200.computercraft.shared.media.items.ItemDiskLegacy; |  | ||||||
| import dan200.computercraft.shared.media.items.ItemPrintout; | import dan200.computercraft.shared.media.items.ItemPrintout; | ||||||
| import dan200.computercraft.shared.media.items.ItemTreasureDisk; | import dan200.computercraft.shared.media.items.ItemTreasureDisk; | ||||||
| import dan200.computercraft.shared.peripheral.common.BlockPeripheral; | import dan200.computercraft.shared.peripheral.diskdrive.BlockDiskDrive; | ||||||
| import dan200.computercraft.shared.peripheral.common.ItemPeripheral; |  | ||||||
| import dan200.computercraft.shared.peripheral.diskdrive.TileDiskDrive; | import dan200.computercraft.shared.peripheral.diskdrive.TileDiskDrive; | ||||||
| import dan200.computercraft.shared.peripheral.modem.wired.*; | import dan200.computercraft.shared.peripheral.modem.wired.*; | ||||||
| import dan200.computercraft.shared.peripheral.modem.wireless.BlockAdvancedModem; | import dan200.computercraft.shared.peripheral.modem.wireless.BlockWirelessModem; | ||||||
| import dan200.computercraft.shared.peripheral.modem.wireless.ItemAdvancedModem; |  | ||||||
| import dan200.computercraft.shared.peripheral.modem.wireless.TileAdvancedModem; |  | ||||||
| import dan200.computercraft.shared.peripheral.modem.wireless.TileWirelessModem; | import dan200.computercraft.shared.peripheral.modem.wireless.TileWirelessModem; | ||||||
|  | import dan200.computercraft.shared.peripheral.monitor.BlockMonitor; | ||||||
| import dan200.computercraft.shared.peripheral.monitor.TileMonitor; | import dan200.computercraft.shared.peripheral.monitor.TileMonitor; | ||||||
|  | import dan200.computercraft.shared.peripheral.printer.BlockPrinter; | ||||||
| import dan200.computercraft.shared.peripheral.printer.TilePrinter; | import dan200.computercraft.shared.peripheral.printer.TilePrinter; | ||||||
|  | import dan200.computercraft.shared.peripheral.speaker.BlockSpeaker; | ||||||
| import dan200.computercraft.shared.peripheral.speaker.TileSpeaker; | import dan200.computercraft.shared.peripheral.speaker.TileSpeaker; | ||||||
| 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; | ||||||
| import dan200.computercraft.shared.turtle.blocks.BlockTurtle; | import dan200.computercraft.shared.turtle.blocks.BlockTurtle; | ||||||
| import dan200.computercraft.shared.turtle.blocks.TileTurtle; | import dan200.computercraft.shared.turtle.blocks.TileTurtle; | ||||||
| import dan200.computercraft.shared.turtle.blocks.TileTurtleAdvanced; | import dan200.computercraft.shared.turtle.core.TurtlePlayer; | ||||||
| import dan200.computercraft.shared.turtle.blocks.TileTurtleExpanded; | import dan200.computercraft.shared.turtle.items.ItemTurtle; | ||||||
| import dan200.computercraft.shared.turtle.items.ItemTurtleAdvanced; |  | ||||||
| import dan200.computercraft.shared.turtle.items.ItemTurtleLegacy; |  | ||||||
| import dan200.computercraft.shared.turtle.items.ItemTurtleNormal; |  | ||||||
| import dan200.computercraft.shared.turtle.upgrades.*; | import dan200.computercraft.shared.turtle.upgrades.*; | ||||||
|  | import dan200.computercraft.shared.util.CreativeTabMain; | ||||||
| import net.minecraft.block.Block; | import net.minecraft.block.Block; | ||||||
|  | import net.minecraft.block.material.Material; | ||||||
|  | import net.minecraft.entity.EntityType; | ||||||
| import net.minecraft.init.Items; | import net.minecraft.init.Items; | ||||||
| import net.minecraft.item.Item; | import net.minecraft.item.Item; | ||||||
| import net.minecraft.item.ItemBlock; | import net.minecraft.item.ItemBlock; | ||||||
|  | import net.minecraft.item.ItemGroup; | ||||||
|  | import net.minecraft.tileentity.TileEntityType; | ||||||
| import net.minecraft.util.ResourceLocation; | import net.minecraft.util.ResourceLocation; | ||||||
| import net.minecraftforge.event.RegistryEvent; | import net.minecraftforge.event.RegistryEvent; | ||||||
|  | import net.minecraftforge.eventbus.api.SubscribeEvent; | ||||||
| import net.minecraftforge.fml.common.Mod; | import net.minecraftforge.fml.common.Mod; | ||||||
| import net.minecraftforge.fml.common.eventhandler.SubscribeEvent; |  | ||||||
| import net.minecraftforge.fml.common.registry.GameRegistry; |  | ||||||
| import net.minecraftforge.registries.IForgeRegistry; | import net.minecraftforge.registries.IForgeRegistry; | ||||||
|  |  | ||||||
| @Mod.EventBusSubscriber( modid = ComputerCraft.MOD_ID ) | @Mod.EventBusSubscriber( modid = ComputerCraft.MOD_ID, bus = Mod.EventBusSubscriber.Bus.MOD ) | ||||||
| public final class Registry | public final class Registry | ||||||
| { | { | ||||||
|  |     private static final ItemGroup mainItemGroup = new CreativeTabMain(); | ||||||
|  |  | ||||||
|     private Registry() |     private Registry() | ||||||
|     { |     { | ||||||
|     } |     } | ||||||
| @@ -64,61 +65,120 @@ public final class Registry | |||||||
|         IForgeRegistry<Block> registry = event.getRegistry(); |         IForgeRegistry<Block> registry = event.getRegistry(); | ||||||
|  |  | ||||||
|         // Computers |         // Computers | ||||||
|         ComputerCraft.Blocks.computer = new BlockComputer(); |         ComputerCraft.Blocks.computerNormal = new BlockComputer( | ||||||
|         ComputerCraft.Blocks.commandComputer = new BlockCommandComputer(); |             Block.Properties.create( Material.ROCK ).hardnessAndResistance( 2.0f ), | ||||||
|  |             ComputerFamily.Normal, TileComputer.FACTORY_NORMAL | ||||||
|         registry.registerAll( |  | ||||||
|             ComputerCraft.Blocks.computer.setRegistryName( new ResourceLocation( ComputerCraft.MOD_ID, "computer" ) ), |  | ||||||
|             ComputerCraft.Blocks.commandComputer.setRegistryName( new ResourceLocation( ComputerCraft.MOD_ID, "command_computer" ) ) |  | ||||||
|         ); |         ); | ||||||
|  |  | ||||||
|         // Turtle |         ComputerCraft.Blocks.computerAdvanced = new BlockComputer( | ||||||
|         ComputerCraft.Blocks.turtle = new BlockTurtle(); |             Block.Properties.create( Material.ROCK ).hardnessAndResistance( 2.0f ), | ||||||
|         ComputerCraft.Blocks.turtleExpanded = new BlockTurtle(); |             ComputerFamily.Advanced, TileComputer.FACTORY_ADVANCED | ||||||
|         ComputerCraft.Blocks.turtleAdvanced = new BlockTurtle(); |         ); | ||||||
|  |  | ||||||
|  |         ComputerCraft.Blocks.computerCommand = new BlockComputer( | ||||||
|  |             Block.Properties.create( Material.ROCK ).hardnessAndResistance( -1, 6000000.0F ), | ||||||
|  |             ComputerFamily.Command, TileCommandComputer.FACTORY | ||||||
|  |         ); | ||||||
|  |  | ||||||
|         registry.registerAll( |         registry.registerAll( | ||||||
|             ComputerCraft.Blocks.turtle.setRegistryName( new ResourceLocation( ComputerCraft.MOD_ID, "turtle" ) ), |             ComputerCraft.Blocks.computerNormal.setRegistryName( new ResourceLocation( ComputerCraft.MOD_ID, "computer_normal" ) ), | ||||||
|             ComputerCraft.Blocks.turtleExpanded.setRegistryName( new ResourceLocation( ComputerCraft.MOD_ID, "turtle_expanded" ) ), |             ComputerCraft.Blocks.computerAdvanced.setRegistryName( new ResourceLocation( ComputerCraft.MOD_ID, "computer_advanced" ) ), | ||||||
|  |             ComputerCraft.Blocks.computerCommand.setRegistryName( new ResourceLocation( ComputerCraft.MOD_ID, "computer_command" ) ) | ||||||
|  |         ); | ||||||
|  |  | ||||||
|  |         // Turtles | ||||||
|  |         ComputerCraft.Blocks.turtleNormal = new BlockTurtle( | ||||||
|  |             Block.Properties.create( Material.ROCK ).hardnessAndResistance( 2.5f ), | ||||||
|  |             ComputerFamily.Normal, TileTurtle.FACTORY_NORMAL | ||||||
|  |         ); | ||||||
|  |  | ||||||
|  |         ComputerCraft.Blocks.turtleAdvanced = new BlockTurtle( | ||||||
|  |             Block.Properties.create( Material.ROCK ).hardnessAndResistance( 2.5f ), | ||||||
|  |             ComputerFamily.Advanced, TileTurtle.FACTORY_ADVANCED | ||||||
|  |         ); | ||||||
|  |  | ||||||
|  |         registry.registerAll( | ||||||
|  |             ComputerCraft.Blocks.turtleNormal.setRegistryName( new ResourceLocation( ComputerCraft.MOD_ID, "turtle_normal" ) ), | ||||||
|             ComputerCraft.Blocks.turtleAdvanced.setRegistryName( new ResourceLocation( ComputerCraft.MOD_ID, "turtle_advanced" ) ) |             ComputerCraft.Blocks.turtleAdvanced.setRegistryName( new ResourceLocation( ComputerCraft.MOD_ID, "turtle_advanced" ) ) | ||||||
|         ); |         ); | ||||||
|  |  | ||||||
|         // Peripheral |         // Peripherals | ||||||
|         ComputerCraft.Blocks.peripheral = new BlockPeripheral(); |         ComputerCraft.Blocks.speaker = new BlockSpeaker( | ||||||
|         registry.register( ComputerCraft.Blocks.peripheral.setRegistryName( new ResourceLocation( ComputerCraft.MOD_ID, "peripheral" ) ) ); |             Block.Properties.create( Material.ROCK ).hardnessAndResistance( 2 ) | ||||||
|  |         ); | ||||||
|  |  | ||||||
|         // Cable |         ComputerCraft.Blocks.diskDrive = new BlockDiskDrive( | ||||||
|         ComputerCraft.Blocks.cable = new BlockCable(); |             Block.Properties.create( Material.ROCK ).hardnessAndResistance( 2 ) | ||||||
|         registry.register( ComputerCraft.Blocks.cable.setRegistryName( new ResourceLocation( ComputerCraft.MOD_ID, "cable" ) ) ); |         ); | ||||||
|  |  | ||||||
|         // Advanced modem |         ComputerCraft.Blocks.monitorNormal = new BlockMonitor( | ||||||
|         ComputerCraft.Blocks.advancedModem = new BlockAdvancedModem(); |             Block.Properties.create( Material.ROCK ).hardnessAndResistance( 2 ), | ||||||
|         registry.register( ComputerCraft.Blocks.advancedModem.setRegistryName( new ResourceLocation( ComputerCraft.MOD_ID, "advanced_modem" ) ) ); |             TileMonitor.FACTORY_NORMAL | ||||||
|  |         ); | ||||||
|  |  | ||||||
|         // Full block modem |         ComputerCraft.Blocks.monitorAdvanced = new BlockMonitor( | ||||||
|         ComputerCraft.Blocks.wiredModemFull = new BlockWiredModemFull(); |             Block.Properties.create( Material.ROCK ).hardnessAndResistance( 2 ), | ||||||
|         registry.register( ComputerCraft.Blocks.wiredModemFull.setRegistryName( new ResourceLocation( ComputerCraft.MOD_ID, "wired_modem_full" ) ) ); |             TileMonitor.FACTORY_ADVANCED | ||||||
|  |         ); | ||||||
|  |  | ||||||
|         registerTileEntities(); |         ComputerCraft.Blocks.printer = new BlockPrinter( | ||||||
|  |             Block.Properties.create( Material.ROCK ).hardnessAndResistance( 2 ) | ||||||
|  |         ); | ||||||
|  |  | ||||||
|  |         ComputerCraft.Blocks.wirelessModemNormal = new BlockWirelessModem( | ||||||
|  |             Block.Properties.create( Material.ROCK ).hardnessAndResistance( 2 ), | ||||||
|  |             TileWirelessModem.FACTORY_NORMAL | ||||||
|  |         ); | ||||||
|  |  | ||||||
|  |         ComputerCraft.Blocks.wirelessModemAdvanced = new BlockWirelessModem( | ||||||
|  |             Block.Properties.create( Material.ROCK ).hardnessAndResistance( 2 ), | ||||||
|  |             TileWirelessModem.FACTORY_ADVANCED | ||||||
|  |         ); | ||||||
|  |  | ||||||
|  |         ComputerCraft.Blocks.wiredModemFull = new BlockWiredModemFull( | ||||||
|  |             Block.Properties.create( Material.ROCK ).hardnessAndResistance( 1.5f ) | ||||||
|  |         ); | ||||||
|  |  | ||||||
|  |         ComputerCraft.Blocks.cable = new BlockCable( | ||||||
|  |             Block.Properties.create( Material.ROCK ).hardnessAndResistance( 1.5f ) | ||||||
|  |         ); | ||||||
|  |  | ||||||
|  |         registry.registerAll( | ||||||
|  |             ComputerCraft.Blocks.speaker.setRegistryName( new ResourceLocation( ComputerCraft.MOD_ID, "speaker" ) ), | ||||||
|  |             ComputerCraft.Blocks.diskDrive.setRegistryName( new ResourceLocation( ComputerCraft.MOD_ID, "disk_drive" ) ), | ||||||
|  |             ComputerCraft.Blocks.monitorNormal.setRegistryName( new ResourceLocation( ComputerCraft.MOD_ID, "monitor_normal" ) ), | ||||||
|  |             ComputerCraft.Blocks.monitorAdvanced.setRegistryName( new ResourceLocation( ComputerCraft.MOD_ID, "monitor_advanced" ) ), | ||||||
|  |             ComputerCraft.Blocks.printer.setRegistryName( new ResourceLocation( ComputerCraft.MOD_ID, "printer" ) ), | ||||||
|  |             ComputerCraft.Blocks.wirelessModemNormal.setRegistryName( new ResourceLocation( ComputerCraft.MOD_ID, "wireless_modem_normal" ) ), | ||||||
|  |             ComputerCraft.Blocks.wirelessModemAdvanced.setRegistryName( new ResourceLocation( ComputerCraft.MOD_ID, "wireless_modem_advanced" ) ), | ||||||
|  |             ComputerCraft.Blocks.wiredModemFull.setRegistryName( new ResourceLocation( ComputerCraft.MOD_ID, "wired_modem_full" ) ), | ||||||
|  |             ComputerCraft.Blocks.cable.setRegistryName( new ResourceLocation( ComputerCraft.MOD_ID, "cable" ) ) | ||||||
|  |         ); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     private static void registerTileEntities() |     @SubscribeEvent | ||||||
|  |     public static void registerTileEntities( RegistryEvent.Register<TileEntityType<?>> event ) | ||||||
|     { |     { | ||||||
|         GameRegistry.registerTileEntity( TileComputer.class, new ResourceLocation( ComputerCraft.MOD_ID, "computer" ) ); |         IForgeRegistry<TileEntityType<?>> registry = event.getRegistry(); | ||||||
|         GameRegistry.registerTileEntity( TileCommandComputer.class, new ResourceLocation( ComputerCraft.MOD_ID, "command_computer" ) ); |  | ||||||
|  |  | ||||||
|         GameRegistry.registerTileEntity( TileTurtle.class, new ResourceLocation( ComputerCraft.MOD_ID, "turtle" ) ); |         // Computers | ||||||
|         GameRegistry.registerTileEntity( TileTurtleExpanded.class, new ResourceLocation( ComputerCraft.MOD_ID, "turtleex" ) ); |         registry.registerAll( TileComputer.FACTORY_NORMAL, TileComputer.FACTORY_ADVANCED, TileCommandComputer.FACTORY ); | ||||||
|         GameRegistry.registerTileEntity( TileTurtleAdvanced.class, new ResourceLocation( ComputerCraft.MOD_ID, "turtleadv" ) ); |  | ||||||
|  |  | ||||||
|         GameRegistry.registerTileEntity( TileDiskDrive.class, new ResourceLocation( ComputerCraft.MOD_ID, "diskdrive" ) ); |         // Turtles | ||||||
|         GameRegistry.registerTileEntity( TileWirelessModem.class, new ResourceLocation( ComputerCraft.MOD_ID, "wirelessmodem" ) ); |         registry.registerAll( TileTurtle.FACTORY_NORMAL, TileTurtle.FACTORY_ADVANCED ); | ||||||
|         GameRegistry.registerTileEntity( TileMonitor.class, new ResourceLocation( ComputerCraft.MOD_ID, "monitor" ) ); |  | ||||||
|         GameRegistry.registerTileEntity( TilePrinter.class, new ResourceLocation( ComputerCraft.MOD_ID, "ccprinter" ) ); |         // Peripherals | ||||||
|         GameRegistry.registerTileEntity( TileCable.class, new ResourceLocation( ComputerCraft.MOD_ID, "wiredmodem" ) ); |         registry.registerAll( | ||||||
|         GameRegistry.registerTileEntity( TileAdvancedModem.class, new ResourceLocation( ComputerCraft.MOD_ID, "advanced_modem" ) ); |             TileSpeaker.FACTORY, | ||||||
|         GameRegistry.registerTileEntity( TileSpeaker.class, new ResourceLocation( ComputerCraft.MOD_ID, "speaker" ) ); |             TileDiskDrive.FACTORY, | ||||||
|         GameRegistry.registerTileEntity( TileWiredModemFull.class, new ResourceLocation( ComputerCraft.MOD_ID, "wired_modem_full" ) ); |             TileMonitor.FACTORY_NORMAL, | ||||||
|  |             TileMonitor.FACTORY_ADVANCED, | ||||||
|  |             TilePrinter.FACTORY, | ||||||
|  |             TileWirelessModem.FACTORY_NORMAL, | ||||||
|  |             TileWirelessModem.FACTORY_ADVANCED, | ||||||
|  |             TileWiredModemFull.FACTORY, | ||||||
|  |             TileCable.FACTORY | ||||||
|  |         ); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     private static <T extends ItemBlock> T setupItemBlock( T item ) |     private static <T extends ItemBlock> T setupItemBlock( T item ) | ||||||
| @@ -127,203 +187,128 @@ public final class Registry | |||||||
|         return item; |         return item; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     private static Item.Properties defaultItem() | ||||||
|  |     { | ||||||
|  |         return new Item.Properties().group( mainItemGroup ); | ||||||
|  |     } | ||||||
|  |  | ||||||
|     @SubscribeEvent |     @SubscribeEvent | ||||||
|     public static void registerItems( RegistryEvent.Register<Item> event ) |     public static void registerItems( RegistryEvent.Register<Item> event ) | ||||||
|     { |     { | ||||||
|         IForgeRegistry<Item> registry = event.getRegistry(); |         IForgeRegistry<Item> registry = event.getRegistry(); | ||||||
|  |  | ||||||
|         // Computers |         // Computer | ||||||
|         ComputerCraft.Items.computer = new ItemComputer( ComputerCraft.Blocks.computer ); |         ComputerCraft.Items.computerNormal = new ItemComputer( ComputerCraft.Blocks.computerNormal, defaultItem() ); | ||||||
|         ComputerCraft.Items.commandComputer = new ItemCommandComputer( ComputerCraft.Blocks.commandComputer ); |         ComputerCraft.Items.computerAdvanced = new ItemComputer( ComputerCraft.Blocks.computerAdvanced, defaultItem() ); | ||||||
|  |         ComputerCraft.Items.computerCommand = new ItemComputer( ComputerCraft.Blocks.computerCommand, defaultItem() ); | ||||||
|  |  | ||||||
|         registry.registerAll( |         registry.registerAll( | ||||||
|             setupItemBlock( ComputerCraft.Items.computer ), |             setupItemBlock( ComputerCraft.Items.computerNormal ), | ||||||
|             setupItemBlock( ComputerCraft.Items.commandComputer ) |             setupItemBlock( ComputerCraft.Items.computerAdvanced ), | ||||||
|         ); |             setupItemBlock( ComputerCraft.Items.computerCommand ) | ||||||
|  |  | ||||||
|         // Pocket computer |  | ||||||
|         ComputerCraft.Items.pocketComputer = new ItemPocketComputer(); |  | ||||||
|         registry.register( |  | ||||||
|             ComputerCraft.Items.pocketComputer.setRegistryName( new ResourceLocation( ComputerCraft.MOD_ID, "pocket_computer" ) ) |  | ||||||
|         ); |         ); | ||||||
|  |  | ||||||
|         // Turtle |         // Turtle | ||||||
|         ComputerCraft.Items.turtle = new ItemTurtleLegacy( ComputerCraft.Blocks.turtle ); |         ComputerCraft.Items.turtleNormal = new ItemTurtle( ComputerCraft.Blocks.turtleNormal, defaultItem() ); | ||||||
|         ComputerCraft.Items.turtleExpanded = new ItemTurtleNormal( ComputerCraft.Blocks.turtleExpanded ); |         ComputerCraft.Items.turtleAdvanced = new ItemTurtle( ComputerCraft.Blocks.turtleAdvanced, defaultItem() ); | ||||||
|         ComputerCraft.Items.turtleAdvanced = new ItemTurtleAdvanced( ComputerCraft.Blocks.turtleAdvanced ); |  | ||||||
|  |  | ||||||
|         registry.registerAll( |         registry.registerAll( | ||||||
|             setupItemBlock( ComputerCraft.Items.turtle ), |             setupItemBlock( ComputerCraft.Items.turtleNormal ), | ||||||
|             setupItemBlock( ComputerCraft.Items.turtleExpanded ), |  | ||||||
|             setupItemBlock( ComputerCraft.Items.turtleAdvanced ) |             setupItemBlock( ComputerCraft.Items.turtleAdvanced ) | ||||||
|         ); |         ); | ||||||
|  |  | ||||||
|         // Printouts |         // Pocket computer | ||||||
|         ComputerCraft.Items.printout = new ItemPrintout(); |         ComputerCraft.Items.pocketComputerNormal = new ItemPocketComputer( defaultItem().maxStackSize( 1 ), ComputerFamily.Normal ); | ||||||
|         registry.register( ComputerCraft.Items.printout.setRegistryName( new ResourceLocation( ComputerCraft.MOD_ID, "printout" ) ) ); |         ComputerCraft.Items.pocketComputerAdvanced = new ItemPocketComputer( defaultItem().maxStackSize( 1 ), ComputerFamily.Advanced ); | ||||||
|  |  | ||||||
|         // Disks |         registry.registerAll( | ||||||
|         ComputerCraft.Items.disk = new ItemDiskLegacy(); |             ComputerCraft.Items.pocketComputerNormal.setRegistryName( new ResourceLocation( ComputerCraft.MOD_ID, "pocket_computer_normal" ) ), | ||||||
|         ComputerCraft.Items.diskExpanded = new ItemDiskExpanded(); |             ComputerCraft.Items.pocketComputerAdvanced.setRegistryName( new ResourceLocation( ComputerCraft.MOD_ID, "pocket_computer_advanced" ) ) | ||||||
|         ComputerCraft.Items.treasureDisk = new ItemTreasureDisk(); |         ); | ||||||
|  |  | ||||||
|  |         // Floppy disk | ||||||
|  |         ComputerCraft.Items.disk = new ItemDisk( defaultItem().maxStackSize( 1 ) ); | ||||||
|  |         ComputerCraft.Items.treasureDisk = new ItemTreasureDisk( defaultItem().maxStackSize( 1 ) ); | ||||||
|  |  | ||||||
|         registry.registerAll( |         registry.registerAll( | ||||||
|             ComputerCraft.Items.disk.setRegistryName( new ResourceLocation( ComputerCraft.MOD_ID, "disk" ) ), |             ComputerCraft.Items.disk.setRegistryName( new ResourceLocation( ComputerCraft.MOD_ID, "disk" ) ), | ||||||
|             ComputerCraft.Items.diskExpanded.setRegistryName( new ResourceLocation( ComputerCraft.MOD_ID, "disk_expanded" ) ), |  | ||||||
|             ComputerCraft.Items.treasureDisk.setRegistryName( new ResourceLocation( ComputerCraft.MOD_ID, "treasure_disk" ) ) |             ComputerCraft.Items.treasureDisk.setRegistryName( new ResourceLocation( ComputerCraft.MOD_ID, "treasure_disk" ) ) | ||||||
|         ); |         ); | ||||||
|  |  | ||||||
|         // Peripherals |         // Printouts | ||||||
|         ComputerCraft.Items.peripheral = new ItemPeripheral( ComputerCraft.Blocks.peripheral ); |         ComputerCraft.Items.printedPage = new ItemPrintout( defaultItem().maxStackSize( 1 ), ItemPrintout.Type.PAGE ); | ||||||
|         ComputerCraft.Items.advancedModem = new ItemAdvancedModem( ComputerCraft.Blocks.advancedModem ); |         ComputerCraft.Items.printedPages = new ItemPrintout( defaultItem().maxStackSize( 1 ), ItemPrintout.Type.PAGES ); | ||||||
|         ComputerCraft.Items.cable = new ItemCable( ComputerCraft.Blocks.cable ); |         ComputerCraft.Items.printedBook = new ItemPrintout( defaultItem().maxStackSize( 1 ), ItemPrintout.Type.BOOK ); | ||||||
|         ComputerCraft.Items.wiredModemFull = new ItemWiredModemFull( ComputerCraft.Blocks.wiredModemFull ); |  | ||||||
|  |  | ||||||
|         registry.registerAll( |         registry.registerAll( | ||||||
|             setupItemBlock( ComputerCraft.Items.peripheral ), |             ComputerCraft.Items.printedPage.setRegistryName( new ResourceLocation( ComputerCraft.MOD_ID, "printed_page" ) ), | ||||||
|             setupItemBlock( ComputerCraft.Items.advancedModem ), |             ComputerCraft.Items.printedPages.setRegistryName( new ResourceLocation( ComputerCraft.MOD_ID, "printed_pages" ) ), | ||||||
|             setupItemBlock( ComputerCraft.Items.cable ), |             ComputerCraft.Items.printedBook.setRegistryName( new ResourceLocation( ComputerCraft.MOD_ID, "printed_book" ) ) | ||||||
|             setupItemBlock( ComputerCraft.Items.wiredModemFull ) |         ); | ||||||
|  |  | ||||||
|  |         // Peripherals | ||||||
|  |         registry.registerAll( | ||||||
|  |             setupItemBlock( new ItemBlock( ComputerCraft.Blocks.speaker, defaultItem() ) ), | ||||||
|  |             setupItemBlock( new ItemBlock( ComputerCraft.Blocks.diskDrive, defaultItem() ) ), | ||||||
|  |             setupItemBlock( new ItemBlock( ComputerCraft.Blocks.printer, defaultItem() ) ), | ||||||
|  |             setupItemBlock( new ItemBlock( ComputerCraft.Blocks.monitorNormal, defaultItem() ) ), | ||||||
|  |             setupItemBlock( new ItemBlock( ComputerCraft.Blocks.monitorAdvanced, defaultItem() ) ), | ||||||
|  |             setupItemBlock( new ItemBlock( ComputerCraft.Blocks.wirelessModemNormal, defaultItem() ) ), | ||||||
|  |             setupItemBlock( new ItemBlock( ComputerCraft.Blocks.wirelessModemAdvanced, defaultItem() ) ), | ||||||
|  |             setupItemBlock( new ItemBlock( ComputerCraft.Blocks.wiredModemFull, defaultItem() ) ) | ||||||
|  |         ); | ||||||
|  |  | ||||||
|  |         ComputerCraft.Items.cable = new ItemBlockCable.Cable( ComputerCraft.Blocks.cable, defaultItem() ); | ||||||
|  |         ComputerCraft.Items.wiredModem = new ItemBlockCable.WiredModem( ComputerCraft.Blocks.cable, defaultItem() ); | ||||||
|  |         registry.registerAll( | ||||||
|  |             ComputerCraft.Items.cable.setRegistryName( new ResourceLocation( ComputerCraft.MOD_ID, "cable" ) ), | ||||||
|  |             ComputerCraft.Items.wiredModem.setRegistryName( new ResourceLocation( ComputerCraft.MOD_ID, "wired_modem" ) ) | ||||||
|         ); |         ); | ||||||
|  |  | ||||||
|         registerTurtleUpgrades(); |         registerTurtleUpgrades(); | ||||||
|         registerPocketUpgrades(); |         registerPocketUpgrades(); | ||||||
|         registerLegacyUpgrades(); |  | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     private static void registerTurtleUpgrades() |     private static void registerTurtleUpgrades() | ||||||
|     { |     { | ||||||
|         // Upgrades |         // Upgrades | ||||||
|         ComputerCraft.TurtleUpgrades.wirelessModem = new TurtleModem( false, new ResourceLocation( "computercraft", "wireless_modem" ), 1 ); |         ComputerCraft.TurtleUpgrades.wirelessModemNormal = new TurtleModem( false, new ResourceLocation( ComputerCraft.MOD_ID, "wireless_modem_normal" ) ); | ||||||
|         TurtleUpgrades.registerInternal( ComputerCraft.TurtleUpgrades.wirelessModem ); |         TurtleUpgrades.register( ComputerCraft.TurtleUpgrades.wirelessModemNormal ); | ||||||
|  |  | ||||||
|         ComputerCraft.TurtleUpgrades.advancedModem = new TurtleModem( true, new ResourceLocation( "computercraft", "advanced_modem" ), -1 ); |         ComputerCraft.TurtleUpgrades.wirelessModemAdvanced = new TurtleModem( true, new ResourceLocation( ComputerCraft.MOD_ID, "wireless_modem_advanced" ) ); | ||||||
|         TurtleUpgrades.registerInternal( ComputerCraft.TurtleUpgrades.advancedModem ); |         ComputerCraftAPI.registerTurtleUpgrade( ComputerCraft.TurtleUpgrades.wirelessModemAdvanced ); | ||||||
|  |  | ||||||
|         ComputerCraft.TurtleUpgrades.speaker = new TurtleSpeaker( new ResourceLocation( "computercraft", "speaker" ), 8 ); |         ComputerCraft.TurtleUpgrades.speaker = new TurtleSpeaker( new ResourceLocation( ComputerCraft.MOD_ID, "speaker" ) ); | ||||||
|         TurtleUpgrades.registerInternal( ComputerCraft.TurtleUpgrades.speaker ); |         ComputerCraftAPI.registerTurtleUpgrade( ComputerCraft.TurtleUpgrades.speaker ); | ||||||
|  |  | ||||||
|         ComputerCraft.TurtleUpgrades.craftingTable = new TurtleCraftingTable( new ResourceLocation( "minecraft", "crafting_table" ), 2 ); |         ComputerCraft.TurtleUpgrades.craftingTable = new TurtleCraftingTable( new ResourceLocation( "minecraft", "crafting_table" ) ); | ||||||
|         TurtleUpgrades.registerInternal( ComputerCraft.TurtleUpgrades.craftingTable ); |         ComputerCraftAPI.registerTurtleUpgrade( ComputerCraft.TurtleUpgrades.craftingTable ); | ||||||
|  |  | ||||||
|         ComputerCraft.TurtleUpgrades.diamondSword = new TurtleSword( new ResourceLocation( "minecraft", "diamond_sword" ), 3, Items.DIAMOND_SWORD ); |         ComputerCraft.TurtleUpgrades.diamondSword = new TurtleSword( new ResourceLocation( "minecraft", "diamond_sword" ), Items.DIAMOND_SWORD ); | ||||||
|         TurtleUpgrades.registerInternal( ComputerCraft.TurtleUpgrades.diamondSword ); |         ComputerCraftAPI.registerTurtleUpgrade( ComputerCraft.TurtleUpgrades.diamondSword ); | ||||||
|  |  | ||||||
|         ComputerCraft.TurtleUpgrades.diamondShovel = new TurtleShovel( new ResourceLocation( "minecraft", "diamond_shovel" ), 4, Items.DIAMOND_SHOVEL ); |         ComputerCraft.TurtleUpgrades.diamondShovel = new TurtleShovel( new ResourceLocation( "minecraft", "diamond_shovel" ), Items.DIAMOND_SHOVEL ); | ||||||
|         TurtleUpgrades.registerInternal( ComputerCraft.TurtleUpgrades.diamondShovel ); |         ComputerCraftAPI.registerTurtleUpgrade( ComputerCraft.TurtleUpgrades.diamondShovel ); | ||||||
|  |  | ||||||
|         ComputerCraft.TurtleUpgrades.diamondPickaxe = new TurtleTool( new ResourceLocation( "minecraft", "diamond_pickaxe" ), 5, Items.DIAMOND_PICKAXE ); |         ComputerCraft.TurtleUpgrades.diamondPickaxe = new TurtleTool( new ResourceLocation( "minecraft", "diamond_pickaxe" ), Items.DIAMOND_PICKAXE ); | ||||||
|         TurtleUpgrades.registerInternal( ComputerCraft.TurtleUpgrades.diamondPickaxe ); |         ComputerCraftAPI.registerTurtleUpgrade( ComputerCraft.TurtleUpgrades.diamondPickaxe ); | ||||||
|  |  | ||||||
|         ComputerCraft.TurtleUpgrades.diamondAxe = new TurtleAxe( new ResourceLocation( "minecraft", "diamond_axe" ), 6, Items.DIAMOND_AXE ); |         ComputerCraft.TurtleUpgrades.diamondAxe = new TurtleAxe( new ResourceLocation( "minecraft", "diamond_axe" ), Items.DIAMOND_AXE ); | ||||||
|         TurtleUpgrades.registerInternal( ComputerCraft.TurtleUpgrades.diamondAxe ); |         ComputerCraftAPI.registerTurtleUpgrade( ComputerCraft.TurtleUpgrades.diamondAxe ); | ||||||
|  |  | ||||||
|         ComputerCraft.TurtleUpgrades.diamondHoe = new TurtleHoe( new ResourceLocation( "minecraft", "diamond_hoe" ), 7, Items.DIAMOND_HOE ); |         ComputerCraft.TurtleUpgrades.diamondHoe = new TurtleHoe( new ResourceLocation( "minecraft", "diamond_hoe" ), Items.DIAMOND_HOE ); | ||||||
|         TurtleUpgrades.registerInternal( ComputerCraft.TurtleUpgrades.diamondHoe ); |         ComputerCraftAPI.registerTurtleUpgrade( ComputerCraft.TurtleUpgrades.diamondHoe ); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     private static void registerPocketUpgrades() |     private static void registerPocketUpgrades() | ||||||
|     { |     { | ||||||
|         // Register pocket upgrades |         ComputerCraftAPI.registerPocketUpgrade( ComputerCraft.PocketUpgrades.wirelessModemNormal = new PocketModem( false ) ); | ||||||
|         ComputerCraft.PocketUpgrades.wirelessModem = new PocketModem( false ); |         ComputerCraftAPI.registerPocketUpgrade( ComputerCraft.PocketUpgrades.wirelessModemAdvanced = new PocketModem( true ) ); | ||||||
|         ComputerCraftAPI.registerPocketUpgrade( ComputerCraft.PocketUpgrades.wirelessModem ); |         ComputerCraftAPI.registerPocketUpgrade( ComputerCraft.PocketUpgrades.speaker = new PocketSpeaker() ); | ||||||
|         ComputerCraft.PocketUpgrades.advancedModem = new PocketModem( true ); |  | ||||||
|         ComputerCraftAPI.registerPocketUpgrade( ComputerCraft.PocketUpgrades.advancedModem ); |  | ||||||
|  |  | ||||||
|         ComputerCraft.PocketUpgrades.speaker = new PocketSpeaker(); |  | ||||||
|         ComputerCraftAPI.registerPocketUpgrade( ComputerCraft.PocketUpgrades.speaker ); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     @SuppressWarnings( "deprecation" ) |  | ||||||
|     private static void registerLegacyUpgrades() |  | ||||||
|     { |  | ||||||
|         ComputerCraft.PocketUpgrades.pocketSpeaker = ComputerCraft.PocketUpgrades.speaker; |  | ||||||
|         ComputerCraft.Upgrades.advancedModem = ComputerCraft.TurtleUpgrades.advancedModem; |  | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @SubscribeEvent |     @SubscribeEvent | ||||||
|     public static void remapItems( RegistryEvent.MissingMappings<Item> mappings ) |     public static void registerEntities( RegistryEvent.Register<EntityType<?>> registry ) | ||||||
|     { |     { | ||||||
|         // We have to use mappings.getAllMappings() as the mod ID is upper case but the domain lower. |         registry.getRegistry().register( TurtlePlayer.TYPE.setRegistryName( new ResourceLocation( ComputerCraft.MOD_ID, "turtle_player" ) ) ); | ||||||
|         for( RegistryEvent.MissingMappings.Mapping<Item> mapping : mappings.getAllMappings() ) |  | ||||||
|         { |  | ||||||
|             String domain = mapping.key.getNamespace(); |  | ||||||
|             if( !domain.equalsIgnoreCase( ComputerCraft.MOD_ID ) ) continue; |  | ||||||
|  |  | ||||||
|             String key = mapping.key.getPath(); |  | ||||||
|             if( key.equalsIgnoreCase( "CC-Computer" ) ) |  | ||||||
|             { |  | ||||||
|                 mapping.remap( ComputerCraft.Items.computer ); |  | ||||||
|             } |  | ||||||
|             else if( key.equalsIgnoreCase( "CC-Peripheral" ) ) |  | ||||||
|             { |  | ||||||
|                 mapping.remap( ComputerCraft.Items.peripheral ); |  | ||||||
|             } |  | ||||||
|             else if( key.equalsIgnoreCase( "CC-Cable" ) ) |  | ||||||
|             { |  | ||||||
|                 mapping.remap( ComputerCraft.Items.cable ); |  | ||||||
|             } |  | ||||||
|             else if( key.equalsIgnoreCase( "diskExpanded" ) ) |  | ||||||
|             { |  | ||||||
|                 mapping.remap( ComputerCraft.Items.diskExpanded ); |  | ||||||
|             } |  | ||||||
|             else if( key.equalsIgnoreCase( "treasureDisk" ) ) |  | ||||||
|             { |  | ||||||
|                 mapping.remap( ComputerCraft.Items.treasureDisk ); |  | ||||||
|             } |  | ||||||
|             else if( key.equalsIgnoreCase( "pocketComputer" ) ) |  | ||||||
|             { |  | ||||||
|                 mapping.remap( ComputerCraft.Items.pocketComputer ); |  | ||||||
|             } |  | ||||||
|             else if( key.equalsIgnoreCase( "CC-Turtle" ) ) |  | ||||||
|             { |  | ||||||
|                 mapping.remap( ComputerCraft.Items.turtle ); |  | ||||||
|             } |  | ||||||
|             else if( key.equalsIgnoreCase( "CC-TurtleExpanded" ) ) |  | ||||||
|             { |  | ||||||
|                 mapping.remap( ComputerCraft.Items.turtleExpanded ); |  | ||||||
|             } |  | ||||||
|             else if( key.equalsIgnoreCase( "CC-TurtleAdvanced" ) ) |  | ||||||
|             { |  | ||||||
|                 mapping.remap( ComputerCraft.Items.turtleAdvanced ); |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     @SubscribeEvent |  | ||||||
|     public static void remapBlocks( RegistryEvent.MissingMappings<Block> mappings ) |  | ||||||
|     { |  | ||||||
|         // We have to use mappings.getAllMappings() as the mod ID is upper case but the domain lower. |  | ||||||
|         for( RegistryEvent.MissingMappings.Mapping<Block> mapping : mappings.getAllMappings() ) |  | ||||||
|         { |  | ||||||
|             String domain = mapping.key.getNamespace(); |  | ||||||
|             if( !domain.equalsIgnoreCase( ComputerCraft.MOD_ID ) ) continue; |  | ||||||
|  |  | ||||||
|             String key = mapping.key.getPath(); |  | ||||||
|             if( key.equalsIgnoreCase( "CC-Computer" ) ) |  | ||||||
|             { |  | ||||||
|                 mapping.remap( ComputerCraft.Blocks.computer ); |  | ||||||
|             } |  | ||||||
|             else if( key.equalsIgnoreCase( "CC-Peripheral" ) ) |  | ||||||
|             { |  | ||||||
|                 mapping.remap( ComputerCraft.Blocks.peripheral ); |  | ||||||
|             } |  | ||||||
|             else if( key.equalsIgnoreCase( "CC-Cable" ) ) |  | ||||||
|             { |  | ||||||
|                 mapping.remap( ComputerCraft.Blocks.cable ); |  | ||||||
|             } |  | ||||||
|             else if( key.equalsIgnoreCase( "CC-Turtle" ) ) |  | ||||||
|             { |  | ||||||
|                 mapping.remap( ComputerCraft.Blocks.turtle ); |  | ||||||
|             } |  | ||||||
|             else if( key.equalsIgnoreCase( "CC-TurtleExpanded" ) ) |  | ||||||
|             { |  | ||||||
|                 mapping.remap( ComputerCraft.Blocks.turtleExpanded ); |  | ||||||
|             } |  | ||||||
|             else if( key.equalsIgnoreCase( "CC-TurtleAdvanced" ) ) |  | ||||||
|             { |  | ||||||
|                 mapping.remap( ComputerCraft.Blocks.turtleAdvanced ); |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
|     } |     } | ||||||
| } | } | ||||||
|   | |||||||
| @@ -6,65 +6,28 @@ | |||||||
|  |  | ||||||
| package dan200.computercraft.shared; | package dan200.computercraft.shared; | ||||||
|  |  | ||||||
| import com.google.common.base.Preconditions; |  | ||||||
| import dan200.computercraft.ComputerCraft; | import dan200.computercraft.ComputerCraft; | ||||||
| import dan200.computercraft.api.permissions.ITurtlePermissionProvider; |  | ||||||
| import dan200.computercraft.api.turtle.event.TurtleActionEvent; | import dan200.computercraft.api.turtle.event.TurtleActionEvent; | ||||||
| import net.minecraft.entity.player.EntityPlayer; | import net.minecraft.entity.player.EntityPlayer; | ||||||
| import net.minecraft.server.MinecraftServer; | import net.minecraft.server.MinecraftServer; | ||||||
| import net.minecraft.util.math.BlockPos; | import net.minecraft.util.math.BlockPos; | ||||||
| import net.minecraft.world.World; | import net.minecraft.world.World; | ||||||
|  | import net.minecraftforge.eventbus.api.SubscribeEvent; | ||||||
| import net.minecraftforge.fml.common.Mod; | import net.minecraftforge.fml.common.Mod; | ||||||
| import net.minecraftforge.fml.common.eventhandler.SubscribeEvent; |  | ||||||
|  |  | ||||||
| import javax.annotation.Nonnull; |  | ||||||
| import java.util.Collection; |  | ||||||
| import java.util.LinkedHashSet; |  | ||||||
|  |  | ||||||
| @Mod.EventBusSubscriber( modid = ComputerCraft.MOD_ID ) | @Mod.EventBusSubscriber( modid = ComputerCraft.MOD_ID ) | ||||||
| public final class TurtlePermissions | public final class TurtlePermissions | ||||||
| { | { | ||||||
|     private static final Collection<ITurtlePermissionProvider> providers = new LinkedHashSet<>(); |  | ||||||
|  |  | ||||||
|     private TurtlePermissions() |  | ||||||
|     { |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     public static void register( @Nonnull ITurtlePermissionProvider upgrade ) |  | ||||||
|     { |  | ||||||
|         Preconditions.checkNotNull( upgrade, "upgrade cannot be null" ); |  | ||||||
|  |  | ||||||
|         providers.add( upgrade ); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     public static boolean isBlockEnterable( World world, BlockPos pos, EntityPlayer player ) |     public static boolean isBlockEnterable( World world, BlockPos pos, EntityPlayer player ) | ||||||
|     { |     { | ||||||
|         MinecraftServer server = player.getServer(); |         MinecraftServer server = world.getServer(); | ||||||
|         if( server != null && !world.isRemote && server.isBlockProtected( world, pos, player ) ) |         return server == null || world.isRemote || !server.isBlockProtected( world, pos, player ); | ||||||
|         { |  | ||||||
|             return false; |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         for( ITurtlePermissionProvider provider : providers ) |  | ||||||
|         { |  | ||||||
|             if( !provider.isBlockEnterable( world, pos ) ) return false; |  | ||||||
|         } |  | ||||||
|         return true; |  | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     public static boolean isBlockEditable( World world, BlockPos pos, EntityPlayer player ) |     public static boolean isBlockEditable( World world, BlockPos pos, EntityPlayer player ) | ||||||
|     { |     { | ||||||
|         MinecraftServer server = player.getServer(); |         MinecraftServer server = world.getServer(); | ||||||
|         if( server != null && !world.isRemote && server.isBlockProtected( world, pos, player ) ) |         return server == null || world.isRemote || !server.isBlockProtected( world, pos, player ); | ||||||
|         { |  | ||||||
|             return false; |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         for( ITurtlePermissionProvider provider : providers ) |  | ||||||
|         { |  | ||||||
|             if( !provider.isBlockEditable( world, pos ) ) return false; |  | ||||||
|         } |  | ||||||
|         return true; |  | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @SubscribeEvent |     @SubscribeEvent | ||||||
|   | |||||||
| @@ -10,11 +10,9 @@ import dan200.computercraft.ComputerCraft; | |||||||
| import dan200.computercraft.api.turtle.ITurtleUpgrade; | import dan200.computercraft.api.turtle.ITurtleUpgrade; | ||||||
| import dan200.computercraft.shared.computer.core.ComputerFamily; | import dan200.computercraft.shared.computer.core.ComputerFamily; | ||||||
| import dan200.computercraft.shared.util.InventoryUtil; | import dan200.computercraft.shared.util.InventoryUtil; | ||||||
| import it.unimi.dsi.fastutil.ints.Int2ObjectMap; |  | ||||||
| import it.unimi.dsi.fastutil.ints.Int2ObjectOpenHashMap; |  | ||||||
| import net.minecraft.item.ItemStack; | import net.minecraft.item.ItemStack; | ||||||
| import net.minecraftforge.fml.common.Loader; | import net.minecraftforge.fml.ModContainer; | ||||||
| import net.minecraftforge.fml.common.ModContainer; | import net.minecraftforge.fml.ModLoadingContext; | ||||||
|  |  | ||||||
| import javax.annotation.Nonnull; | import javax.annotation.Nonnull; | ||||||
| import javax.annotation.Nullable; | import javax.annotation.Nullable; | ||||||
| @@ -23,7 +21,6 @@ import java.util.*; | |||||||
| public final class TurtleUpgrades | public final class TurtleUpgrades | ||||||
| { | { | ||||||
|     private static final Map<String, ITurtleUpgrade> upgrades = new HashMap<>(); |     private static final Map<String, ITurtleUpgrade> upgrades = new HashMap<>(); | ||||||
|     private static final Int2ObjectMap<ITurtleUpgrade> legacyUpgrades = new Int2ObjectOpenHashMap<>(); |  | ||||||
|     private static final IdentityHashMap<ITurtleUpgrade, String> upgradeOwners = new IdentityHashMap<>(); |     private static final IdentityHashMap<ITurtleUpgrade, String> upgradeOwners = new IdentityHashMap<>(); | ||||||
|  |  | ||||||
|     private TurtleUpgrades() {} |     private TurtleUpgrades() {} | ||||||
| @@ -32,73 +29,25 @@ public final class TurtleUpgrades | |||||||
|     { |     { | ||||||
|         Objects.requireNonNull( upgrade, "upgrade cannot be null" ); |         Objects.requireNonNull( upgrade, "upgrade cannot be null" ); | ||||||
|  |  | ||||||
|         int id = upgrade.getLegacyUpgradeID(); |  | ||||||
|         if( id >= 0 && id < 64 ) |  | ||||||
|         { |  | ||||||
|             String message = getMessage( upgrade, "Legacy UpgradeID '" + id + "' is reserved by ComputerCraft" ); |  | ||||||
|             ComputerCraft.log.error( message ); |  | ||||||
|             throw new RuntimeException( message ); |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         registerInternal( upgrade ); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     static void registerInternal( ITurtleUpgrade upgrade ) |  | ||||||
|     { |  | ||||||
|         Objects.requireNonNull( upgrade, "upgrade cannot be null" ); |  | ||||||
|  |  | ||||||
|         // Check conditions |  | ||||||
|         int legacyId = upgrade.getLegacyUpgradeID(); |  | ||||||
|         if( legacyId >= 0 ) |  | ||||||
|         { |  | ||||||
|             if( legacyId >= Short.MAX_VALUE ) |  | ||||||
|             { |  | ||||||
|                 String message = getMessage( upgrade, "UpgradeID '" + legacyId + "' is out of range" ); |  | ||||||
|                 ComputerCraft.log.error( message ); |  | ||||||
|                 throw new RuntimeException( message ); |  | ||||||
|             } |  | ||||||
|  |  | ||||||
|             ITurtleUpgrade existing = legacyUpgrades.get( legacyId ); |  | ||||||
|             if( existing != null ) |  | ||||||
|             { |  | ||||||
|                 String message = getMessage( upgrade, "UpgradeID '" + legacyId + "' is already registered by '" + existing.getUnlocalisedAdjective() + " Turtle'" ); |  | ||||||
|                 ComputerCraft.log.error( message ); |  | ||||||
|                 throw new RuntimeException( message ); |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         String id = upgrade.getUpgradeID().toString(); |         String id = upgrade.getUpgradeID().toString(); | ||||||
|         ITurtleUpgrade existing = upgrades.get( id ); |         ITurtleUpgrade existing = upgrades.get( id ); | ||||||
|         if( existing != null ) |         if( existing != null ) | ||||||
|         { |         { | ||||||
|             String message = getMessage( upgrade, "UpgradeID '" + id + "' is already registered by '" + existing.getUnlocalisedAdjective() + " Turtle'" ); |             throw new IllegalStateException( "Error registering '" + upgrade.getUnlocalisedAdjective() + " Turle'. UpgradeID '" + id + "' is already registered by '" + existing.getUnlocalisedAdjective() + " Turtle'" ); | ||||||
|             ComputerCraft.log.error( message ); |  | ||||||
|             throw new RuntimeException( message ); |  | ||||||
|         } |         } | ||||||
|  |  | ||||||
|         // Register |  | ||||||
|         if( legacyId >= 0 ) legacyUpgrades.put( legacyId, upgrade ); |  | ||||||
|         upgrades.put( id, upgrade ); |         upgrades.put( id, upgrade ); | ||||||
|  |  | ||||||
|         ModContainer mc = Loader.instance().activeModContainer(); |         ModContainer mc = ModLoadingContext.get().getActiveContainer(); | ||||||
|         if( mc != null && mc.getModId() != null ) upgradeOwners.put( upgrade, mc.getModId() ); |         if( mc != null && mc.getModId() != null ) upgradeOwners.put( upgrade, mc.getModId() ); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     private static String getMessage( ITurtleUpgrade upgrade, String rest ) |  | ||||||
|     { |  | ||||||
|         return "Error registering '" + upgrade.getUnlocalisedAdjective() + " Turtle'. " + rest; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     public static ITurtleUpgrade get( String id ) |     public static ITurtleUpgrade get( String id ) | ||||||
|     { |     { | ||||||
|         return upgrades.get( id ); |         return upgrades.get( id ); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     public static ITurtleUpgrade get( int id ) |  | ||||||
|     { |  | ||||||
|         return legacyUpgrades.get( id ); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     public static ITurtleUpgrade get( @Nonnull ItemStack stack ) |     public static ITurtleUpgrade get( @Nonnull ItemStack stack ) | ||||||
|     { |     { | ||||||
|         if( stack.isEmpty() ) return null; |         if( stack.isEmpty() ) return null; | ||||||
| @@ -118,15 +67,20 @@ public final class TurtleUpgrades | |||||||
|     public static Iterable<ITurtleUpgrade> getVanillaUpgrades() |     public static Iterable<ITurtleUpgrade> getVanillaUpgrades() | ||||||
|     { |     { | ||||||
|         List<ITurtleUpgrade> vanilla = new ArrayList<>(); |         List<ITurtleUpgrade> vanilla = new ArrayList<>(); | ||||||
|  |  | ||||||
|  |  | ||||||
|  |         // ComputerCraft upgrades | ||||||
|  |         vanilla.add( ComputerCraft.TurtleUpgrades.wirelessModemNormal ); | ||||||
|  |         vanilla.add( ComputerCraft.TurtleUpgrades.wirelessModemAdvanced ); | ||||||
|  |         vanilla.add( ComputerCraft.TurtleUpgrades.speaker ); | ||||||
|  |  | ||||||
|  |         // Vanilla Minecraft upgrades | ||||||
|         vanilla.add( ComputerCraft.TurtleUpgrades.diamondPickaxe ); |         vanilla.add( ComputerCraft.TurtleUpgrades.diamondPickaxe ); | ||||||
|         vanilla.add( ComputerCraft.TurtleUpgrades.diamondAxe ); |         vanilla.add( ComputerCraft.TurtleUpgrades.diamondAxe ); | ||||||
|         vanilla.add( ComputerCraft.TurtleUpgrades.diamondSword ); |         vanilla.add( ComputerCraft.TurtleUpgrades.diamondSword ); | ||||||
|         vanilla.add( ComputerCraft.TurtleUpgrades.diamondShovel ); |         vanilla.add( ComputerCraft.TurtleUpgrades.diamondShovel ); | ||||||
|         vanilla.add( ComputerCraft.TurtleUpgrades.diamondHoe ); |         vanilla.add( ComputerCraft.TurtleUpgrades.diamondHoe ); | ||||||
|         vanilla.add( ComputerCraft.TurtleUpgrades.craftingTable ); |         vanilla.add( ComputerCraft.TurtleUpgrades.craftingTable ); | ||||||
|         vanilla.add( ComputerCraft.TurtleUpgrades.wirelessModem ); |  | ||||||
|         vanilla.add( ComputerCraft.TurtleUpgrades.advancedModem ); |  | ||||||
|         vanilla.add( ComputerCraft.TurtleUpgrades.speaker ); |  | ||||||
|         return vanilla; |         return vanilla; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|   | |||||||
| @@ -6,7 +6,9 @@ | |||||||
|  |  | ||||||
| package dan200.computercraft.shared.command; | package dan200.computercraft.shared.command; | ||||||
|  |  | ||||||
| import com.google.common.collect.Sets; | import com.mojang.brigadier.CommandDispatcher; | ||||||
|  | import com.mojang.brigadier.arguments.StringArgumentType; | ||||||
|  | import com.mojang.brigadier.exceptions.CommandSyntaxException; | ||||||
| import dan200.computercraft.ComputerCraft; | import dan200.computercraft.ComputerCraft; | ||||||
| import dan200.computercraft.api.peripheral.IPeripheral; | import dan200.computercraft.api.peripheral.IPeripheral; | ||||||
| import dan200.computercraft.core.apis.IAPIEnvironment; | import dan200.computercraft.core.apis.IAPIEnvironment; | ||||||
| @@ -15,30 +17,37 @@ import dan200.computercraft.core.tracking.ComputerTracker; | |||||||
| import dan200.computercraft.core.tracking.Tracking; | import dan200.computercraft.core.tracking.Tracking; | ||||||
| import dan200.computercraft.core.tracking.TrackingContext; | import dan200.computercraft.core.tracking.TrackingContext; | ||||||
| import dan200.computercraft.core.tracking.TrackingField; | import dan200.computercraft.core.tracking.TrackingField; | ||||||
| import dan200.computercraft.shared.Config; |  | ||||||
| import dan200.computercraft.shared.command.framework.*; |  | ||||||
| import dan200.computercraft.shared.command.text.TableBuilder; | import dan200.computercraft.shared.command.text.TableBuilder; | ||||||
| import dan200.computercraft.shared.computer.core.ComputerFamily; | import dan200.computercraft.shared.computer.core.ComputerFamily; | ||||||
| import dan200.computercraft.shared.computer.core.ServerComputer; | import dan200.computercraft.shared.computer.core.ServerComputer; | ||||||
| import dan200.computercraft.shared.network.Containers; | import dan200.computercraft.shared.network.Containers; | ||||||
| import net.minecraft.command.CommandBase; | import net.minecraft.command.CommandSource; | ||||||
| import net.minecraft.command.CommandException; |  | ||||||
| import net.minecraft.command.ICommandSender; |  | ||||||
| import net.minecraft.entity.Entity; | import net.minecraft.entity.Entity; | ||||||
|  | import net.minecraft.entity.player.EntityPlayer; | ||||||
| import net.minecraft.entity.player.EntityPlayerMP; | import net.minecraft.entity.player.EntityPlayerMP; | ||||||
| import net.minecraft.server.MinecraftServer; | import net.minecraft.network.play.server.SPacketPlayerPosLook; | ||||||
| import net.minecraft.util.math.BlockPos; | import net.minecraft.util.math.BlockPos; | ||||||
| import net.minecraft.util.text.ITextComponent; | import net.minecraft.util.text.ITextComponent; | ||||||
| import net.minecraft.util.text.TextComponentString; | import net.minecraft.util.text.TextComponentString; | ||||||
| import net.minecraft.world.World; | import net.minecraft.world.World; | ||||||
|  | import net.minecraft.world.WorldServer; | ||||||
|  |  | ||||||
| import javax.annotation.Nonnull; | import javax.annotation.Nonnull; | ||||||
| import java.util.*; | import java.util.*; | ||||||
| import java.util.function.Consumer; |  | ||||||
|  |  | ||||||
|  | import static dan200.computercraft.shared.command.CommandUtils.isPlayer; | ||||||
|  | import static dan200.computercraft.shared.command.Exceptions.*; | ||||||
|  | import static dan200.computercraft.shared.command.arguments.ComputerArgumentType.getComputerArgument; | ||||||
|  | import static dan200.computercraft.shared.command.arguments.ComputerArgumentType.oneComputer; | ||||||
|  | import static dan200.computercraft.shared.command.arguments.ComputersArgumentType.*; | ||||||
|  | import static dan200.computercraft.shared.command.arguments.TrackingFieldArgumentType.trackingField; | ||||||
|  | import static dan200.computercraft.shared.command.builder.CommandBuilder.args; | ||||||
|  | import static dan200.computercraft.shared.command.builder.CommandBuilder.command; | ||||||
|  | import static dan200.computercraft.shared.command.builder.HelpingArgumentBuilder.choice; | ||||||
| import static dan200.computercraft.shared.command.text.ChatHelpers.*; | import static dan200.computercraft.shared.command.text.ChatHelpers.*; | ||||||
|  | import static net.minecraft.command.Commands.literal; | ||||||
|  |  | ||||||
| public final class CommandComputerCraft extends CommandDelegate | public final class CommandComputerCraft | ||||||
| { | { | ||||||
|     public static final UUID SYSTEM_UUID = new UUID( 0, 0 ); |     public static final UUID SYSTEM_UUID = new UUID( 0, 0 ); | ||||||
|  |  | ||||||
| @@ -46,335 +55,209 @@ public final class CommandComputerCraft extends CommandDelegate | |||||||
|     private static final int DUMP_SINGLE_ID = 1844510720; |     private static final int DUMP_SINGLE_ID = 1844510720; | ||||||
|     private static final int TRACK_ID = 373882880; |     private static final int TRACK_ID = 373882880; | ||||||
|  |  | ||||||
|     public CommandComputerCraft() |     private CommandComputerCraft() | ||||||
|     { |     { | ||||||
|         super( create() ); |  | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     private static ISubCommand create() |     public static void register( CommandDispatcher<CommandSource> dispatcher ) | ||||||
|     { |     { | ||||||
|         CommandRoot root = new CommandRoot( "computercraft" ); |         dispatcher.register( choice( "computercraft" ) | ||||||
|  |             .then( literal( "dump" ) | ||||||
|         root.register( new SubCommandBase( "dump", UserLevel.OWNER_OP ) |                 .requires( UserLevel.OWNER_OP ) | ||||||
|         { |                 .executes( context -> { | ||||||
|             @Override |  | ||||||
|             public void execute( @Nonnull CommandContext context, @Nonnull List<String> arguments ) throws CommandException |  | ||||||
|             { |  | ||||||
|                 if( arguments.isEmpty() ) |  | ||||||
|                 { |  | ||||||
|                     TableBuilder table = new TableBuilder( DUMP_LIST_ID, "Computer", "On", "Position" ); |                     TableBuilder table = new TableBuilder( DUMP_LIST_ID, "Computer", "On", "Position" ); | ||||||
|  |  | ||||||
|  |                     CommandSource source = context.getSource(); | ||||||
|                     List<ServerComputer> computers = new ArrayList<>( ComputerCraft.serverComputerRegistry.getComputers() ); |                     List<ServerComputer> computers = new ArrayList<>( ComputerCraft.serverComputerRegistry.getComputers() ); | ||||||
|  |  | ||||||
|                     // Unless we're on a server, limit the number of rows we can send. |                     // Unless we're on a server, limit the number of rows we can send. | ||||||
|                     if( !(context.getSender() instanceof MinecraftServer) ) |                     World world = source.getWorld(); | ||||||
|                     { |                     BlockPos pos = new BlockPos( source.getPos() ); | ||||||
|                         World world = context.getSender().getEntityWorld(); |  | ||||||
|                         BlockPos pos = context.getSender().getPosition(); |  | ||||||
|  |  | ||||||
|                         computers.sort( ( a, b ) -> { |                     computers.sort( ( a, b ) -> { | ||||||
|                             if( a.getWorld() == b.getWorld() && a.getWorld() == world ) |                         if( a.getWorld() == b.getWorld() && a.getWorld() == world ) | ||||||
|                             { |                         { | ||||||
|                                 return Double.compare( a.getPosition().distanceSq( pos ), b.getPosition().distanceSq( pos ) ); |                             return Double.compare( a.getPosition().distanceSq( pos ), b.getPosition().distanceSq( pos ) ); | ||||||
|                             } |                         } | ||||||
|                             else if( a.getWorld() == world ) |                         else if( a.getWorld() == world ) | ||||||
|                             { |                         { | ||||||
|                                 return -1; |                             return -1; | ||||||
|                             } |                         } | ||||||
|                             else if( b.getWorld() == world ) |                         else if( b.getWorld() == world ) | ||||||
|                             { |                         { | ||||||
|                                 return 1; |                             return 1; | ||||||
|                             } |                         } | ||||||
|                             else |                         else | ||||||
|                             { |                         { | ||||||
|                                 return Integer.compare( a.getInstanceID(), b.getInstanceID() ); |                             return Integer.compare( a.getInstanceID(), b.getInstanceID() ); | ||||||
|                             } |                         } | ||||||
|                         } ); |                     } ); | ||||||
|                     } |  | ||||||
|  |  | ||||||
|                     for( ServerComputer computer : computers ) |                     for( ServerComputer computer : computers ) | ||||||
|                     { |                     { | ||||||
|                         table.row( |                         table.row( | ||||||
|                             linkComputer( context, computer, computer.getID() ), |                             linkComputer( source, computer, computer.getID() ), | ||||||
|                             bool( computer.isOn() ), |                             bool( computer.isOn() ), | ||||||
|                             linkPosition( context, computer ) |                             linkPosition( source, computer ) | ||||||
|                         ); |                         ); | ||||||
|                     } |                     } | ||||||
|  |  | ||||||
|                     table.display( context.getSender() ); |                     table.display( context.getSource() ); | ||||||
|                 } |                     return computers.size(); | ||||||
|                 else if( arguments.size() == 1 ) |                 } ) | ||||||
|                 { |                 .then( args() | ||||||
|                     ServerComputer computer = ComputerSelector.getComputer( arguments.get( 0 ) ); |                     .arg( "computer", oneComputer() ) | ||||||
|  |                     .executes( context -> { | ||||||
|  |                         ServerComputer computer = getComputerArgument( context, "computer" ); | ||||||
|  |  | ||||||
|                     TableBuilder table = new TableBuilder( DUMP_SINGLE_ID ); |                         TableBuilder table = new TableBuilder( DUMP_SINGLE_ID ); | ||||||
|                     table.row( header( "Instance" ), text( Integer.toString( computer.getInstanceID() ) ) ); |                         table.row( header( "Instance" ), text( Integer.toString( computer.getInstanceID() ) ) ); | ||||||
|                     table.row( header( "Id" ), text( Integer.toString( computer.getID() ) ) ); |                         table.row( header( "Id" ), text( Integer.toString( computer.getID() ) ) ); | ||||||
|                     table.row( header( "Label" ), text( computer.getLabel() ) ); |                         table.row( header( "Label" ), text( computer.getLabel() ) ); | ||||||
|                     table.row( header( "On" ), bool( computer.isOn() ) ); |                         table.row( header( "On" ), bool( computer.isOn() ) ); | ||||||
|                     table.row( header( "Position" ), linkPosition( context, computer ) ); |                         table.row( header( "Position" ), linkPosition( context.getSource(), computer ) ); | ||||||
|                     table.row( header( "Family" ), text( computer.getFamily().toString() ) ); |                         table.row( header( "Family" ), text( computer.getFamily().toString() ) ); | ||||||
|  |  | ||||||
|                     for( int i = 0; i < 6; i++ ) |                         for( int i = 0; i < 6; i++ ) | ||||||
|                     { |  | ||||||
|                         IPeripheral peripheral = computer.getPeripheral( i ); |  | ||||||
|                         if( peripheral != null ) |  | ||||||
|                         { |                         { | ||||||
|                             table.row( header( "Peripheral " + IAPIEnvironment.SIDE_NAMES[i] ), text( peripheral.getType() ) ); |                             IPeripheral peripheral = computer.getPeripheral( i ); | ||||||
|  |                             if( peripheral != null ) | ||||||
|  |                             { | ||||||
|  |                                 table.row( header( "Peripheral " + IAPIEnvironment.SIDE_NAMES[i] ), text( peripheral.getType() ) ); | ||||||
|  |                             } | ||||||
|                         } |                         } | ||||||
|                     } |  | ||||||
|  |  | ||||||
|                     table.display( context.getSender() ); |                         table.display( context.getSource() ); | ||||||
|                 } |                         return 1; | ||||||
|                 else |                     } ) ) ) | ||||||
|                 { |  | ||||||
|                     throw new CommandException( context.getFullUsage() ); |  | ||||||
|                 } |  | ||||||
|             } |  | ||||||
|  |  | ||||||
|             @Nonnull |             .then( command( "shutdown" ) | ||||||
|             @Override |                 .requires( UserLevel.OWNER_OP ) | ||||||
|             public List<String> getCompletion( @Nonnull CommandContext context, @Nonnull List<String> arguments ) |                 .argManyValue( "computers", manyComputers(), s -> ComputerCraft.serverComputerRegistry.getComputers() ) | ||||||
|             { |                 .executes( ( context, computers ) -> { | ||||||
|                 return arguments.size() == 1 |  | ||||||
|                     ? ComputerSelector.completeComputer( arguments.get( 0 ) ) |  | ||||||
|                     : Collections.emptyList(); |  | ||||||
|             } |  | ||||||
|         } ); |  | ||||||
|  |  | ||||||
|         root.register( new SubCommandBase( "shutdown", UserLevel.OWNER_OP ) |  | ||||||
|         { |  | ||||||
|             @Override |  | ||||||
|             public void execute( @Nonnull CommandContext context, @Nonnull List<String> arguments ) throws CommandException |  | ||||||
|             { |  | ||||||
|                 withComputers( arguments, computers -> { |  | ||||||
|                     int shutdown = 0; |                     int shutdown = 0; | ||||||
|                     for( ServerComputer computer : computers ) |                     for( ServerComputer computer : unwrap( context.getSource(), computers ) ) | ||||||
|                     { |                     { | ||||||
|                         if( computer.isOn() ) shutdown++; |                         if( computer.isOn() ) shutdown++; | ||||||
|                         computer.shutdown(); |                         computer.shutdown(); | ||||||
|                     } |                     } | ||||||
|                     context.getSender().sendMessage( translate( "commands.computercraft.shutdown.done", shutdown, computers.size() ) ); |                     context.getSource().sendFeedback( translate( "commands.computercraft.shutdown.done", shutdown, computers.size() ), false ); | ||||||
|                 } ); |                     return shutdown; | ||||||
|             } |                 } ) ) | ||||||
|  |  | ||||||
|             @Nonnull |             .then( command( "turn-on" ) | ||||||
|             @Override |                 .requires( UserLevel.OWNER_OP ) | ||||||
|             public List<String> getCompletion( @Nonnull CommandContext context, @Nonnull List<String> arguments ) |                 .argManyValue( "computers", manyComputers(), s -> ComputerCraft.serverComputerRegistry.getComputers() ) | ||||||
|             { |                 .executes( ( context, computers ) -> { | ||||||
|                 return arguments.isEmpty() |  | ||||||
|                     ? Collections.emptyList() |  | ||||||
|                     : ComputerSelector.completeComputer( arguments.get( arguments.size() - 1 ) ); |  | ||||||
|             } |  | ||||||
|         } ); |  | ||||||
|  |  | ||||||
|         root.register( new SubCommandBase( "turn-on", UserLevel.OWNER_OP ) |  | ||||||
|         { |  | ||||||
|             @Override |  | ||||||
|             public void execute( @Nonnull CommandContext context, @Nonnull List<String> arguments ) throws CommandException |  | ||||||
|             { |  | ||||||
|                 withComputers( arguments, computers -> { |  | ||||||
|                     int on = 0; |                     int on = 0; | ||||||
|                     for( ServerComputer computer : computers ) |                     for( ServerComputer computer : unwrap( context.getSource(), computers ) ) | ||||||
|                     { |                     { | ||||||
|                         if( !computer.isOn() ) on++; |                         if( !computer.isOn() ) on++; | ||||||
|                         computer.turnOn(); |                         computer.turnOn(); | ||||||
|                     } |                     } | ||||||
|                     context.getSender().sendMessage( translate( "commands.computercraft.turn_on.done", on, computers.size() ) ); |                     context.getSource().sendFeedback( translate( "commands.computercraft.turn_on.done", on, computers.size() ), false ); | ||||||
|                 } ); |                     return on; | ||||||
|             } |                 } ) ) | ||||||
|  |  | ||||||
|             @Nonnull |             .then( command( "tp" ) | ||||||
|             @Override |                 .requires( UserLevel.OP ) | ||||||
|             public List<String> getCompletion( @Nonnull CommandContext context, @Nonnull List<String> arguments ) |                 .arg( "computer", oneComputer() ) | ||||||
|             { |                 .executes( context -> { | ||||||
|                 return arguments.isEmpty() |                     ServerComputer computer = getComputerArgument( context, "computer" ); | ||||||
|                     ? Collections.emptyList() |                     World world = computer.getWorld(); | ||||||
|                     : ComputerSelector.completeComputer( arguments.get( arguments.size() - 1 ) ); |                     BlockPos pos = computer.getPosition(); | ||||||
|             } |  | ||||||
|         } ); |  | ||||||
|  |  | ||||||
|         root.register( new SubCommandBase( "tp", UserLevel.OP ) |                     if( world == null || pos == null ) throw TP_NOT_THERE.create(); | ||||||
|         { |  | ||||||
|             @Override |  | ||||||
|             public void execute( @Nonnull CommandContext context, @Nonnull List<String> arguments ) throws CommandException |  | ||||||
|             { |  | ||||||
|                 if( arguments.size() != 1 ) throw new CommandException( context.getFullUsage() ); |  | ||||||
|  |  | ||||||
|                 ServerComputer computer = ComputerSelector.getComputer( arguments.get( 0 ) ); |                     Entity entity = context.getSource().assertIsEntity(); | ||||||
|                 World world = computer.getWorld(); |                     if( !(entity instanceof EntityPlayerMP) ) throw TP_NOT_PLAYER.create(); | ||||||
|                 BlockPos pos = computer.getPosition(); |  | ||||||
|  |  | ||||||
|                 if( world == null || pos == null ) throw new CommandException( "commands.computercraft.tp.not_there" ); |                     EntityPlayerMP player = (EntityPlayerMP) entity; | ||||||
|  |                     if( player.getEntityWorld() == world ) | ||||||
|                 ICommandSender sender = context.getSender(); |  | ||||||
|                 if( !(sender instanceof Entity) ) throw new CommandException( "commands.computercraft.tp.not_entity" ); |  | ||||||
|  |  | ||||||
|                 if( sender instanceof EntityPlayerMP ) |  | ||||||
|                 { |  | ||||||
|                     EntityPlayerMP entity = (EntityPlayerMP) sender; |  | ||||||
|                     if( entity.getEntityWorld() != world ) |  | ||||||
|                     { |                     { | ||||||
|                         context.getServer().getPlayerList().changePlayerDimension( entity, world.provider.getDimension() ); |                         player.connection.setPlayerLocation( pos.getX() + 0.5, pos.getY(), pos.getZ() + 0.5, 0, 0, EnumSet.noneOf( SPacketPlayerPosLook.EnumFlags.class ) ); | ||||||
|  |                     } | ||||||
|  |                     else | ||||||
|  |                     { | ||||||
|  |                         player.teleport( (WorldServer) world, pos.getX() + 0.5, pos.getY(), pos.getZ() + 0.5, 0, 0 ); | ||||||
|                     } |                     } | ||||||
|  |  | ||||||
|                     entity.setPositionAndUpdate( pos.getX() + 0.5, pos.getY(), pos.getZ() + 0.5 ); |                     return 1; | ||||||
|                 } |                 } ) ) | ||||||
|                 else |  | ||||||
|                 { |             .then( command( "queue" ) | ||||||
|                     Entity entity = (Entity) sender; |                 .requires( UserLevel.ANYONE ) | ||||||
|                     if( entity.getEntityWorld() != world ) |                 .arg( "computer", manyComputers() ) | ||||||
|  |                 .argManyValue( "args", StringArgumentType.string(), Collections.emptyList() ) | ||||||
|  |                 .executes( ( ctx, args ) -> { | ||||||
|  |                     Collection<ServerComputer> computers = getComputersArgument( ctx, "computer" ); | ||||||
|  |                     Object[] rest = args.toArray(); | ||||||
|  |  | ||||||
|  |                     int queued = 0; | ||||||
|  |                     for( ServerComputer computer : computers ) | ||||||
|                     { |                     { | ||||||
|                         entity.changeDimension( world.provider.getDimension() ); |                         if( computer.getFamily() == ComputerFamily.Command && computer.isOn() ) | ||||||
|  |                         { | ||||||
|  |                             computer.queueEvent( "computer_command", rest ); | ||||||
|  |                             queued++; | ||||||
|  |                         } | ||||||
|                     } |                     } | ||||||
|  |  | ||||||
|                     entity.setLocationAndAngles( |                     return queued; | ||||||
|                         pos.getX() + 0.5, pos.getY(), pos.getZ() + 0.5, |                 } ) ) | ||||||
|                         entity.rotationYaw, entity.rotationPitch |  | ||||||
|                     ); |  | ||||||
|                 } |  | ||||||
|             } |  | ||||||
|  |  | ||||||
|             @Nonnull |             .then( command( "view" ) | ||||||
|             @Override |                 .requires( UserLevel.OP ) | ||||||
|             public List<String> getCompletion( @Nonnull CommandContext context, @Nonnull List<String> arguments ) |                 .arg( "computer", oneComputer() ) | ||||||
|             { |                 .executes( context -> { | ||||||
|                 return arguments.size() == 1 |                     EntityPlayerMP player = context.getSource().asPlayer(); | ||||||
|                     ? ComputerSelector.completeComputer( arguments.get( 0 ) ) |                     ServerComputer computer = getComputerArgument( context, "computer" ); | ||||||
|                     : Collections.emptyList(); |                     Containers.openComputerGUI( player, computer ); | ||||||
|             } |                     return 1; | ||||||
|         } ); |                 } ) ) | ||||||
|  |  | ||||||
|         root.register( new SubCommandBase( "view", UserLevel.OP ) |             .then( choice( "track" ) | ||||||
|         { |                 .then( command( "start" ) | ||||||
|             @Override |                     .requires( UserLevel.OWNER_OP ) | ||||||
|             public void execute( @Nonnull CommandContext context, @Nonnull List<String> arguments ) throws CommandException |                     .executes( context -> { | ||||||
|             { |                         getTimingContext( context.getSource() ).start(); | ||||||
|                 if( arguments.size() != 1 ) throw new CommandException( context.getFullUsage() ); |  | ||||||
|  |  | ||||||
|                 ICommandSender sender = context.getSender(); |                         String stopCommand = "/computercraft track stop"; | ||||||
|                 if( !(sender instanceof EntityPlayerMP) ) |                         context.getSource().sendFeedback( translate( "commands.computercraft.track.start.stop", | ||||||
|                 { |                             link( text( stopCommand ), stopCommand, translate( "commands.computercraft.track.stop.action" ) ) ), false ); | ||||||
|                     throw new CommandException( "commands.computercraft.view.not_player" ); |                         return 1; | ||||||
|                 } |                     } ) ) | ||||||
|  |  | ||||||
|                 ServerComputer computer = ComputerSelector.getComputer( arguments.get( 0 ) ); |                 .then( command( "stop" ) | ||||||
|                 Containers.openComputerGUI( (EntityPlayerMP) sender, computer ); |                     .requires( UserLevel.OWNER_OP ) | ||||||
|             } |                     .executes( context -> { | ||||||
|  |                         TrackingContext timings = getTimingContext( context.getSource() ); | ||||||
|  |                         if( !timings.stop() ) throw NOT_TRACKING_EXCEPTION.create(); | ||||||
|  |                         displayTimings( context.getSource(), timings.getImmutableTimings(), TrackingField.AVERAGE_TIME, DEFAULT_FIELDS ); | ||||||
|  |                         return 1; | ||||||
|  |                     } ) ) | ||||||
|  |  | ||||||
|             @Nonnull |                 .then( command( "dump" ) | ||||||
|             @Override |                     .requires( UserLevel.OWNER_OP ) | ||||||
|             public List<String> getCompletion( @Nonnull CommandContext context, @Nonnull List<String> arguments ) |                     .argManyValue( "fields", trackingField(), DEFAULT_FIELDS ) | ||||||
|             { |                     .executes( ( context, fields ) -> { | ||||||
|                 return arguments.size() == 1 |                         TrackingField sort; | ||||||
|                     ? ComputerSelector.completeComputer( arguments.get( 0 ) ) |                         if( fields.size() == 1 && DEFAULT_FIELDS.contains( fields.get( 0 ) ) ) | ||||||
|                     : Collections.emptyList(); |                         { | ||||||
|             } |                             sort = fields.get( 0 ); | ||||||
|         } ); |                             fields = DEFAULT_FIELDS; | ||||||
|  |                         } | ||||||
|  |                         else | ||||||
|  |                         { | ||||||
|  |                             sort = fields.get( 0 ); | ||||||
|  |                         } | ||||||
|  |  | ||||||
|         root.register( new CommandRoot( "track" ).register( new SubCommandBase( "start", UserLevel.OWNER_OP ) |                         return displayTimings( context.getSource(), sort, fields ); | ||||||
|         { |                     } ) ) ) | ||||||
|             @Override |         ); | ||||||
|             public void execute( @Nonnull CommandContext context, @Nonnull List<String> arguments ) |  | ||||||
|             { |  | ||||||
|                 getTimingContext( context ).start(); |  | ||||||
|  |  | ||||||
|                 String stopCommand = "/" + context.parent().getFullPath() + " stop"; |  | ||||||
|                 context.getSender().sendMessage( list( |  | ||||||
|                     translate( "commands.computercraft.track.start.stop", |  | ||||||
|                         link( text( stopCommand ), stopCommand, translate( "commands.computercraft.track.stop.action" ) ) ) |  | ||||||
|                 ) ); |  | ||||||
|             } |  | ||||||
|         } ).register( new SubCommandBase( "stop", UserLevel.OWNER_OP ) |  | ||||||
|         { |  | ||||||
|             @Override |  | ||||||
|             public void execute( @Nonnull CommandContext context, @Nonnull List<String> arguments ) throws CommandException |  | ||||||
|             { |  | ||||||
|                 TrackingContext timings = getTimingContext( context ); |  | ||||||
|                 if( !timings.stop() ) throw new CommandException( "commands.computercraft.track.stop.not_enabled" ); |  | ||||||
|                 displayTimings( context, timings.getImmutableTimings(), TrackingField.AVERAGE_TIME ); |  | ||||||
|             } |  | ||||||
|         } ).register( new SubCommandBase( "dump", UserLevel.OWNER_OP ) |  | ||||||
|         { |  | ||||||
|             @Override |  | ||||||
|             public void execute( @Nonnull CommandContext context, @Nonnull List<String> arguments ) throws CommandException |  | ||||||
|             { |  | ||||||
|                 TrackingField field = TrackingField.AVERAGE_TIME; |  | ||||||
|                 if( arguments.size() >= 1 ) |  | ||||||
|                 { |  | ||||||
|                     field = TrackingField.fields().get( arguments.get( 0 ) ); |  | ||||||
|                     if( field == null ) |  | ||||||
|                     { |  | ||||||
|                         throw new CommandException( "commands.computercraft.track.dump.no_field", arguments.get( 0 ) ); |  | ||||||
|                     } |  | ||||||
|                 } |  | ||||||
|  |  | ||||||
|                 displayTimings( context, getTimingContext( context ).getImmutableTimings(), field ); |  | ||||||
|             } |  | ||||||
|  |  | ||||||
|             @Nonnull |  | ||||||
|             @Override |  | ||||||
|             public List<String> getCompletion( @Nonnull CommandContext context, @Nonnull List<String> arguments ) |  | ||||||
|             { |  | ||||||
|                 if( arguments.size() == 1 ) |  | ||||||
|                 { |  | ||||||
|                     String match = arguments.get( 0 ); |  | ||||||
|  |  | ||||||
|                     List<String> out = new ArrayList<>(); |  | ||||||
|                     for( String key : TrackingField.fields().keySet() ) |  | ||||||
|                     { |  | ||||||
|                         if( CommandBase.doesStringStartWith( match, key ) ) out.add( key ); |  | ||||||
|                     } |  | ||||||
|  |  | ||||||
|                     out.sort( Comparator.naturalOrder() ); |  | ||||||
|                     return out; |  | ||||||
|                 } |  | ||||||
|                 else |  | ||||||
|                 { |  | ||||||
|                     return super.getCompletion( context, arguments ); |  | ||||||
|                 } |  | ||||||
|             } |  | ||||||
|         } ) ); |  | ||||||
|  |  | ||||||
|         root.register( new SubCommandBase( "reload", UserLevel.OWNER_OP ) |  | ||||||
|         { |  | ||||||
|             @Override |  | ||||||
|             public void execute( @Nonnull CommandContext context, @Nonnull List<String> arguments ) |  | ||||||
|             { |  | ||||||
|                 Config.reload(); |  | ||||||
|                 context.getSender().sendMessage( translate( "commands.computercraft.reload.done" ) ); |  | ||||||
|             } |  | ||||||
|         } ); |  | ||||||
|  |  | ||||||
|         root.register( new SubCommandBase( "queue", UserLevel.ANYONE ) |  | ||||||
|         { |  | ||||||
|             @Override |  | ||||||
|             public void execute( @Nonnull CommandContext context, @Nonnull List<String> arguments ) throws CommandException |  | ||||||
|             { |  | ||||||
|                 if( arguments.size() < 1 ) throw new CommandException( context.getFullUsage() ); |  | ||||||
|  |  | ||||||
|                 String selector = arguments.get( 0 ); |  | ||||||
|                 Object[] rest = arguments.subList( 1, arguments.size() ).toArray(); |  | ||||||
|  |  | ||||||
|                 boolean found = false; |  | ||||||
|                 for( ServerComputer computer : ComputerSelector.getComputers( selector ) ) |  | ||||||
|                 { |  | ||||||
|                     if( computer.getFamily() != ComputerFamily.Command || !computer.isOn() ) continue; |  | ||||||
|                     found = true; |  | ||||||
|                     computer.queueEvent( "computer_command", rest ); |  | ||||||
|                 } |  | ||||||
|  |  | ||||||
|                 if( !found ) |  | ||||||
|                 { |  | ||||||
|                     throw new CommandException( "commands.computercraft.argument.no_matching", selector ); |  | ||||||
|                 } |  | ||||||
|             } |  | ||||||
|         } ); |  | ||||||
|  |  | ||||||
|         return root; |  | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     private static ITextComponent linkComputer( CommandContext context, ServerComputer serverComputer, int computerId ) |     private static ITextComponent linkComputer( CommandSource source, ServerComputer serverComputer, int computerId ) | ||||||
|     { |     { | ||||||
|         ITextComponent out = new TextComponentString( "" ); |         ITextComponent out = new TextComponentString( "" ); | ||||||
|  |  | ||||||
| @@ -396,7 +279,7 @@ public final class CommandComputerCraft extends CommandDelegate | |||||||
|         out.appendText( " (id " + computerId + ")" ); |         out.appendText( " (id " + computerId + ")" ); | ||||||
|  |  | ||||||
|         // And, if we're a player, some useful links |         // And, if we're a player, some useful links | ||||||
|         if( serverComputer != null && UserLevel.OP.canExecute( context ) && context.fromPlayer() ) |         if( serverComputer != null && UserLevel.OP.test( source ) && isPlayer( source ) ) | ||||||
|         { |         { | ||||||
|             out |             out | ||||||
|                 .appendText( " " ) |                 .appendText( " " ) | ||||||
| @@ -416,9 +299,9 @@ public final class CommandComputerCraft extends CommandDelegate | |||||||
|         return out; |         return out; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     private static ITextComponent linkPosition( CommandContext context, ServerComputer computer ) |     private static ITextComponent linkPosition( CommandSource context, ServerComputer computer ) | ||||||
|     { |     { | ||||||
|         if( UserLevel.OP.canExecute( context ) ) |         if( UserLevel.OP.test( context ) ) | ||||||
|         { |         { | ||||||
|             return link( |             return link( | ||||||
|                 position( computer.getPosition() ), |                 position( computer.getPosition() ), | ||||||
| @@ -432,22 +315,23 @@ public final class CommandComputerCraft extends CommandDelegate | |||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     private static TrackingContext getTimingContext( CommandContext context ) |     @Nonnull | ||||||
|  |     private static TrackingContext getTimingContext( CommandSource source ) | ||||||
|     { |     { | ||||||
|         Entity entity = context.getSender().getCommandSenderEntity(); |         Entity entity = source.getEntity(); | ||||||
|         if( entity instanceof EntityPlayerMP ) |         return entity instanceof EntityPlayer ? Tracking.getContext( entity.getUniqueID() ) : Tracking.getContext( SYSTEM_UUID ); | ||||||
|         { |  | ||||||
|             return Tracking.getContext( entity.getUniqueID() ); |  | ||||||
|         } |  | ||||||
|         else |  | ||||||
|         { |  | ||||||
|             return Tracking.getContext( SYSTEM_UUID ); |  | ||||||
|         } |  | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     private static void displayTimings( CommandContext context, List<ComputerTracker> timings, TrackingField field ) throws CommandException |     private static final List<TrackingField> DEFAULT_FIELDS = Arrays.asList( TrackingField.TASKS, TrackingField.TOTAL_TIME, TrackingField.AVERAGE_TIME, TrackingField.MAX_TIME ); | ||||||
|  |  | ||||||
|  |     private static int displayTimings( CommandSource source, TrackingField sortField, List<TrackingField> fields ) throws CommandSyntaxException | ||||||
|     { |     { | ||||||
|         if( timings.isEmpty() ) throw new CommandException( "commands.computercraft.track.dump.no_timings" ); |         return displayTimings( source, getTimingContext( source ).getTimings(), sortField, fields ); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     private static int displayTimings( CommandSource source, @Nonnull List<ComputerTracker> timings, @Nonnull TrackingField sortField, @Nonnull List<TrackingField> fields ) throws CommandSyntaxException | ||||||
|  |     { | ||||||
|  |         if( timings.isEmpty() ) throw NO_TIMINGS_EXCEPTION.create(); | ||||||
|  |  | ||||||
|         Map<Computer, ServerComputer> lookup = new HashMap<>(); |         Map<Computer, ServerComputer> lookup = new HashMap<>(); | ||||||
|         int maxId = 0, maxInstance = 0; |         int maxId = 0, maxInstance = 0; | ||||||
| @@ -459,74 +343,27 @@ public final class CommandComputerCraft extends CommandDelegate | |||||||
|             if( server.getID() > maxId ) maxId = server.getID(); |             if( server.getID() > maxId ) maxId = server.getID(); | ||||||
|         } |         } | ||||||
|  |  | ||||||
|         timings.sort( Comparator.<ComputerTracker, Long>comparing( x -> x.get( field ) ).reversed() ); |         timings.sort( Comparator.<ComputerTracker, Long>comparing( x -> x.get( sortField ) ).reversed() ); | ||||||
|  |  | ||||||
|         boolean defaultLayout = field == TrackingField.TASKS || field == TrackingField.TOTAL_TIME |         ITextComponent[] headers = new ITextComponent[1 + fields.size()]; | ||||||
|             || field == TrackingField.AVERAGE_TIME || field == TrackingField.MAX_TIME; |         headers[0] = translate( "commands.computercraft.track.dump.computer" ); | ||||||
|  |         for( int i = 0; i < fields.size(); i++ ) headers[i + 1] = translate( fields.get( i ).translationKey() ); | ||||||
|  |         TableBuilder table = new TableBuilder( TRACK_ID, headers ); | ||||||
|         TableBuilder table = defaultLayout ? new TableBuilder( |  | ||||||
|             TRACK_ID, |  | ||||||
|             translate( "commands.computercraft.track.dump.computer" ), |  | ||||||
|             translate( TrackingField.TASKS.translationKey() ), |  | ||||||
|             translate( TrackingField.TOTAL_TIME.translationKey() ), |  | ||||||
|             translate( TrackingField.AVERAGE_TIME.translationKey() ), |  | ||||||
|             translate( TrackingField.MAX_TIME.translationKey() ) |  | ||||||
|         ) : new TableBuilder( |  | ||||||
|             TRACK_ID, |  | ||||||
|             translate( "commands.computercraft.track.dump.computer" ), |  | ||||||
|             translate( field.translationKey() ) |  | ||||||
|         ); |  | ||||||
|  |  | ||||||
|         for( ComputerTracker entry : timings ) |         for( ComputerTracker entry : timings ) | ||||||
|         { |         { | ||||||
|             Computer computer = entry.getComputer(); |             Computer computer = entry.getComputer(); | ||||||
|             ServerComputer serverComputer = computer == null ? null : lookup.get( computer ); |             ServerComputer serverComputer = computer == null ? null : lookup.get( computer ); | ||||||
|  |  | ||||||
|             ITextComponent computerComponent = linkComputer( context, serverComputer, entry.getComputerId() ); |             ITextComponent computerComponent = linkComputer( source, serverComputer, entry.getComputerId() ); | ||||||
|  |  | ||||||
|             if( defaultLayout ) |             ITextComponent[] row = new ITextComponent[1 + fields.size()]; | ||||||
|             { |             row[0] = computerComponent; | ||||||
|                 table.row( |             for( int i = 0; i < fields.size(); i++ ) row[i + 1] = text( entry.getFormatted( fields.get( i ) ) ); | ||||||
|                     computerComponent, |             table.row( row ); | ||||||
|                     text( entry.getFormatted( TrackingField.TASKS ) ), |  | ||||||
|                     text( entry.getFormatted( TrackingField.TOTAL_TIME ) ), |  | ||||||
|                     text( entry.getFormatted( TrackingField.AVERAGE_TIME ) ), |  | ||||||
|                     text( entry.getFormatted( TrackingField.MAX_TIME ) ) |  | ||||||
|                 ); |  | ||||||
|             } |  | ||||||
|             else |  | ||||||
|             { |  | ||||||
|                 table.row( computerComponent, text( entry.getFormatted( field ) ) ); |  | ||||||
|             } |  | ||||||
|         } |         } | ||||||
|  |  | ||||||
|         table.display( context.getSender() ); |         table.display( source ); | ||||||
|     } |         return timings.size(); | ||||||
|  |  | ||||||
|     private static void withComputers( List<String> selectors, Consumer<Collection<ServerComputer>> action ) throws CommandException |  | ||||||
|     { |  | ||||||
|         Set<ServerComputer> computers = Sets.newHashSet(); |  | ||||||
|         List<String> failed = new ArrayList<>(); |  | ||||||
|         if( selectors.isEmpty() ) |  | ||||||
|         { |  | ||||||
|             computers.addAll( ComputerCraft.serverComputerRegistry.getComputers() ); |  | ||||||
|         } |  | ||||||
|         else |  | ||||||
|         { |  | ||||||
|             for( String selector : selectors ) |  | ||||||
|             { |  | ||||||
|                 List<ServerComputer> selected = ComputerSelector.getComputers( selector ); |  | ||||||
|                 computers.addAll( selected ); |  | ||||||
|                 if( selected.isEmpty() ) failed.add( selector ); |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         action.accept( computers ); |  | ||||||
|  |  | ||||||
|         if( !failed.isEmpty() ) |  | ||||||
|         { |  | ||||||
|             throw new CommandException( "commands.computercraft.argument.no_matching", String.join( ", ", failed ) ); |  | ||||||
|         } |  | ||||||
|     } |     } | ||||||
| } | } | ||||||
|   | |||||||
| @@ -6,73 +6,61 @@ | |||||||
|  |  | ||||||
| package dan200.computercraft.shared.command; | package dan200.computercraft.shared.command; | ||||||
|  |  | ||||||
| import net.minecraft.client.gui.GuiScreen; | import com.mojang.brigadier.CommandDispatcher; | ||||||
| import net.minecraft.command.CommandBase; | import com.mojang.brigadier.arguments.StringArgumentType; | ||||||
| import net.minecraft.command.ICommandSender; | import dan200.computercraft.ComputerCraft; | ||||||
| import net.minecraft.server.MinecraftServer; | import net.minecraft.client.Minecraft; | ||||||
|  | import net.minecraft.command.CommandSource; | ||||||
| import net.minecraft.util.text.ITextComponent; | import net.minecraft.util.text.ITextComponent; | ||||||
| import net.minecraft.util.text.TextComponentString; | import net.minecraft.util.text.TextComponentString; | ||||||
| import net.minecraft.util.text.TextComponentTranslation; | import net.minecraft.util.text.TextComponentTranslation; | ||||||
| import net.minecraft.util.text.event.ClickEvent; | import net.minecraft.util.text.event.ClickEvent; | ||||||
| import net.minecraft.util.text.event.HoverEvent; | import net.minecraft.util.text.event.HoverEvent; | ||||||
| import net.minecraftforge.client.IClientCommand; | import net.minecraftforge.api.distmarker.Dist; | ||||||
| import net.minecraftforge.fml.relauncher.Side; | import net.minecraftforge.client.event.ClientChatEvent; | ||||||
| import net.minecraftforge.fml.relauncher.SideOnly; | import net.minecraftforge.eventbus.api.SubscribeEvent; | ||||||
|  | import net.minecraftforge.fml.common.Mod; | ||||||
|  |  | ||||||
| import javax.annotation.Nonnull; | import static net.minecraft.command.Commands.argument; | ||||||
|  | import static net.minecraft.command.Commands.literal; | ||||||
|  |  | ||||||
| public final class CommandCopy extends CommandBase implements IClientCommand | @Mod.EventBusSubscriber( modid = ComputerCraft.MOD_ID, value = Dist.CLIENT ) | ||||||
|  | public final class CommandCopy | ||||||
| { | { | ||||||
|     public static final CommandCopy INSTANCE = new CommandCopy(); |     private static final String PREFIX = "/computercraft copy "; | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * We start with a "~" so we're less likely to show up on completions. |  | ||||||
|      */ |  | ||||||
|     private static final String NAME = "~computercraft_copy"; |  | ||||||
|  |  | ||||||
|     private CommandCopy() |     private CommandCopy() | ||||||
|     { |     { | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @Override |     public static void register( CommandDispatcher<CommandSource> registry ) | ||||||
|     public boolean allowUsageWithoutPrefix( ICommandSender sender, String message ) |  | ||||||
|     { |     { | ||||||
|         return false; |         registry.register( literal( "computercraft" ) | ||||||
|  |             .then( literal( "copy" ) ) | ||||||
|  |             .then( argument( "message", StringArgumentType.greedyString() ) ) | ||||||
|  |             .executes( context -> { | ||||||
|  |                 Minecraft.getInstance().keyboardListener.setClipboardString( context.getArgument( "message", String.class ) ); | ||||||
|  |                 return 1; | ||||||
|  |             } ) | ||||||
|  |         ); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @Nonnull |     @SubscribeEvent | ||||||
|     @Override |     public static void onClientSendMessage( ClientChatEvent event ) | ||||||
|     public String getName() |  | ||||||
|     { |     { | ||||||
|         return NAME; |         // Emulate the command on the client side | ||||||
|     } |         if( event.getMessage().startsWith( PREFIX ) ) | ||||||
|  |         { | ||||||
|     @Nonnull |             Minecraft.getInstance().keyboardListener.setClipboardString( event.getMessage().substring( PREFIX.length() ) ); | ||||||
|     @Override |             event.setCanceled( true ); | ||||||
|     public String getUsage( @Nonnull ICommandSender sender ) |         } | ||||||
|     { |  | ||||||
|         return "/" + NAME + " <text>"; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     @Override |  | ||||||
|     public int getRequiredPermissionLevel() |  | ||||||
|     { |  | ||||||
|         return 0; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     @Override |  | ||||||
|     @SideOnly( Side.CLIENT ) |  | ||||||
|     public void execute( @Nonnull MinecraftServer server, @Nonnull ICommandSender sender, @Nonnull String[] args ) |  | ||||||
|     { |  | ||||||
|         String message = String.join( " ", args ); |  | ||||||
|         if( !message.isEmpty() ) GuiScreen.setClipboardString( message ); |  | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     public static ITextComponent createCopyText( String text ) |     public static ITextComponent createCopyText( String text ) | ||||||
|     { |     { | ||||||
|         TextComponentString name = new TextComponentString( text ); |         TextComponentString name = new TextComponentString( text ); | ||||||
|         name.getStyle() |         name.getStyle() | ||||||
|             .setClickEvent( new ClickEvent( ClickEvent.Action.RUN_COMMAND, "/" + NAME + " " + text ) ) |             .setClickEvent( new ClickEvent( ClickEvent.Action.RUN_COMMAND, PREFIX + text ) ) | ||||||
|             .setHoverEvent( new HoverEvent( HoverEvent.Action.SHOW_TEXT, new TextComponentTranslation( "gui.computercraft.tooltip.copy" ) ) ); |             .setHoverEvent( new HoverEvent( HoverEvent.Action.SHOW_TEXT, new TextComponentTranslation( "gui.computercraft.tooltip.copy" ) ) ); | ||||||
|         return name; |         return name; | ||||||
|     } |     } | ||||||
|   | |||||||
| @@ -6,18 +6,65 @@ | |||||||
|  |  | ||||||
| package dan200.computercraft.shared.command; | package dan200.computercraft.shared.command; | ||||||
|  |  | ||||||
| import net.minecraft.command.ICommandSender; | import com.mojang.brigadier.context.CommandContext; | ||||||
|  | import com.mojang.brigadier.suggestion.Suggestions; | ||||||
|  | import com.mojang.brigadier.suggestion.SuggestionsBuilder; | ||||||
|  | import net.minecraft.command.CommandSource; | ||||||
|  | import net.minecraft.command.ISuggestionProvider; | ||||||
|  | import net.minecraft.entity.Entity; | ||||||
| import net.minecraft.entity.player.EntityPlayerMP; | import net.minecraft.entity.player.EntityPlayerMP; | ||||||
| import net.minecraftforge.common.util.FakePlayer; | import net.minecraftforge.common.util.FakePlayer; | ||||||
|  |  | ||||||
|  | import java.util.Arrays; | ||||||
|  | import java.util.Locale; | ||||||
|  | import java.util.concurrent.CompletableFuture; | ||||||
|  | import java.util.function.Function; | ||||||
|  |  | ||||||
| public final class CommandUtils | public final class CommandUtils | ||||||
| { | { | ||||||
|     private CommandUtils() {} |     private CommandUtils() {} | ||||||
|  |  | ||||||
|     public static boolean isPlayer( ICommandSender sender ) |     public static boolean isPlayer( CommandSource output ) | ||||||
|     { |     { | ||||||
|  |         Entity sender = output.getEntity(); | ||||||
|         return sender instanceof EntityPlayerMP |         return sender instanceof EntityPlayerMP | ||||||
|             && !(sender instanceof FakePlayer) |             && !(sender instanceof FakePlayer) | ||||||
|             && ((EntityPlayerMP) sender).connection != null; |             && ((EntityPlayerMP) sender).connection != null; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     @SuppressWarnings( "unchecked" ) | ||||||
|  |     public static CompletableFuture<Suggestions> suggestOnServer( CommandContext<?> context, SuggestionsBuilder builder, Function<CommandContext<CommandSource>, CompletableFuture<Suggestions>> supplier ) | ||||||
|  |     { | ||||||
|  |         Object source = context.getSource(); | ||||||
|  |         if( !(source instanceof ISuggestionProvider) ) | ||||||
|  |         { | ||||||
|  |             return Suggestions.empty(); | ||||||
|  |         } | ||||||
|  |         else if( source instanceof CommandSource ) | ||||||
|  |         { | ||||||
|  |             return supplier.apply( (CommandContext<CommandSource>) context ); | ||||||
|  |         } | ||||||
|  |         else | ||||||
|  |         { | ||||||
|  |             return ((ISuggestionProvider) source).getSuggestionsFromServer( (CommandContext<ISuggestionProvider>) context, builder ); | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     public static <T> CompletableFuture<Suggestions> suggest( SuggestionsBuilder builder, Iterable<T> candidates, Function<T, String> toString ) | ||||||
|  |     { | ||||||
|  |         String remaining = builder.getRemaining().toLowerCase( Locale.ROOT ); | ||||||
|  |         for( T choice : candidates ) | ||||||
|  |         { | ||||||
|  |             String name = toString.apply( choice ); | ||||||
|  |             if( !name.toLowerCase( Locale.ROOT ).startsWith( remaining ) ) continue; | ||||||
|  |             builder.suggest( name ); | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         return builder.buildFuture(); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     public static <T> CompletableFuture<Suggestions> suggest( SuggestionsBuilder builder, T[] candidates, Function<T, String> toString ) | ||||||
|  |     { | ||||||
|  |         return suggest( builder, Arrays.asList( candidates ), toString ); | ||||||
|  |     } | ||||||
| } | } | ||||||
|   | |||||||
| @@ -1,163 +0,0 @@ | |||||||
| /* |  | ||||||
|  * This file is part of ComputerCraft - http://www.computercraft.info |  | ||||||
|  * Copyright Daniel Ratcliffe, 2011-2019. Do not distribute without permission. |  | ||||||
|  * Send enquiries to dratcliffe@gmail.com |  | ||||||
|  */ |  | ||||||
|  |  | ||||||
| package dan200.computercraft.shared.command; |  | ||||||
|  |  | ||||||
| import com.google.common.collect.Lists; |  | ||||||
| import com.google.common.collect.Sets; |  | ||||||
| import dan200.computercraft.ComputerCraft; |  | ||||||
| import dan200.computercraft.shared.computer.core.ComputerFamily; |  | ||||||
| import dan200.computercraft.shared.computer.core.ServerComputer; |  | ||||||
| import net.minecraft.command.CommandException; |  | ||||||
|  |  | ||||||
| import java.util.*; |  | ||||||
| import java.util.function.Predicate; |  | ||||||
|  |  | ||||||
| final class ComputerSelector |  | ||||||
| { |  | ||||||
|     private ComputerSelector() {} |  | ||||||
|  |  | ||||||
|     private static List<ServerComputer> getComputers( Predicate<ServerComputer> predicate ) |  | ||||||
|     { |  | ||||||
|         // We copy it to prevent concurrent modifications. |  | ||||||
|         ArrayList<ServerComputer> computers = new ArrayList<>( ComputerCraft.serverComputerRegistry.getComputers() ); |  | ||||||
|         computers.removeIf( predicate.negate() ); |  | ||||||
|         return computers; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     static ServerComputer getComputer( String selector ) throws CommandException |  | ||||||
|     { |  | ||||||
|         List<ServerComputer> computers = getComputers( selector ); |  | ||||||
|         if( computers.isEmpty() ) |  | ||||||
|         { |  | ||||||
|             throw new CommandException( "commands.computercraft.argument.no_matching", selector ); |  | ||||||
|         } |  | ||||||
|         else if( computers.size() == 1 ) |  | ||||||
|         { |  | ||||||
|             return computers.get( 0 ); |  | ||||||
|         } |  | ||||||
|         else |  | ||||||
|         { |  | ||||||
|             StringBuilder builder = new StringBuilder(); |  | ||||||
|  |  | ||||||
|             for( int i = 0; i < computers.size(); i++ ) |  | ||||||
|             { |  | ||||||
|                 if( i > 0 ) builder.append( ", " ); |  | ||||||
|                 builder.append( computers.get( i ).getInstanceID() ); |  | ||||||
|             } |  | ||||||
|  |  | ||||||
|             throw new CommandException( "commands.computercraft.argument.many_matching", selector, builder.toString() ); |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     static List<ServerComputer> getComputers( String selector ) throws CommandException |  | ||||||
|     { |  | ||||||
|         if( !selector.isEmpty() && selector.charAt( 0 ) == '#' ) |  | ||||||
|         { |  | ||||||
|             selector = selector.substring( 1 ); |  | ||||||
|  |  | ||||||
|             int id; |  | ||||||
|             try |  | ||||||
|             { |  | ||||||
|                 id = Integer.parseInt( selector ); |  | ||||||
|             } |  | ||||||
|             catch( NumberFormatException e ) |  | ||||||
|             { |  | ||||||
|                 throw new CommandException( "commands.computercraft.argument.not_number", selector ); |  | ||||||
|             } |  | ||||||
|  |  | ||||||
|             return getComputers( x -> x.getID() == id ); |  | ||||||
|         } |  | ||||||
|         else if( !selector.isEmpty() && selector.charAt( 0 ) == '@' ) |  | ||||||
|         { |  | ||||||
|             String label = selector.substring( 1 ); |  | ||||||
|             return getComputers( x -> Objects.equals( label, x.getLabel() ) ); |  | ||||||
|         } |  | ||||||
|         else if( !selector.isEmpty() && selector.charAt( 0 ) == '~' ) |  | ||||||
|         { |  | ||||||
|             String familyName = selector.substring( 1 ); |  | ||||||
|             return getComputers( x -> x.getFamily().name().equalsIgnoreCase( familyName ) ); |  | ||||||
|         } |  | ||||||
|         else |  | ||||||
|         { |  | ||||||
|             int instance; |  | ||||||
|             try |  | ||||||
|             { |  | ||||||
|                 instance = Integer.parseInt( selector ); |  | ||||||
|             } |  | ||||||
|             catch( NumberFormatException e ) |  | ||||||
|             { |  | ||||||
|                 throw new CommandException( "commands.computercraft.argument.not_number", selector ); |  | ||||||
|             } |  | ||||||
|  |  | ||||||
|             ServerComputer computer = ComputerCraft.serverComputerRegistry.get( instance ); |  | ||||||
|             return computer == null ? Collections.emptyList() : Collections.singletonList( computer ); |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     static List<String> completeComputer( String selector ) |  | ||||||
|     { |  | ||||||
|         TreeSet<String> options = Sets.newTreeSet(); |  | ||||||
|  |  | ||||||
|         // We copy it to prevent concurrent modifications. |  | ||||||
|         List<ServerComputer> computers = Lists.newArrayList( ComputerCraft.serverComputerRegistry.getComputers() ); |  | ||||||
|  |  | ||||||
|         if( !selector.isEmpty() && selector.charAt( 0 ) == '#' ) |  | ||||||
|         { |  | ||||||
|             selector = selector.substring( 1 ); |  | ||||||
|  |  | ||||||
|             for( ServerComputer computer : computers ) |  | ||||||
|             { |  | ||||||
|                 String id = Integer.toString( computer.getID() ); |  | ||||||
|                 if( id.startsWith( selector ) ) options.add( "#" + id ); |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
|         else if( !selector.isEmpty() && selector.charAt( 0 ) == '@' ) |  | ||||||
|         { |  | ||||||
|             String label = selector.substring( 1 ); |  | ||||||
|             for( ServerComputer computer : computers ) |  | ||||||
|             { |  | ||||||
|                 String thisLabel = computer.getLabel(); |  | ||||||
|                 if( thisLabel != null && thisLabel.startsWith( label ) ) options.add( "@" + thisLabel ); |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
|         else if( !selector.isEmpty() && selector.charAt( 0 ) == '~' ) |  | ||||||
|         { |  | ||||||
|             String familyName = selector.substring( 1 ).toLowerCase( Locale.ENGLISH ); |  | ||||||
|             for( ComputerFamily family : ComputerFamily.values() ) |  | ||||||
|             { |  | ||||||
|                 if( family.name().toLowerCase( Locale.ENGLISH ).startsWith( familyName ) ) |  | ||||||
|                 { |  | ||||||
|                     options.add( "~" + family.name() ); |  | ||||||
|                 } |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
|         else |  | ||||||
|         { |  | ||||||
|             for( ServerComputer computer : computers ) |  | ||||||
|             { |  | ||||||
|                 String id = Integer.toString( computer.getInstanceID() ); |  | ||||||
|                 if( id.startsWith( selector ) ) options.add( id ); |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         if( options.size() > 100 ) |  | ||||||
|         { |  | ||||||
|             ArrayList<String> result = Lists.newArrayListWithCapacity( 100 ); |  | ||||||
|             for( String element : options ) |  | ||||||
|             { |  | ||||||
|                 if( result.size() > 100 ) break; |  | ||||||
|                 result.add( element ); |  | ||||||
|             } |  | ||||||
|  |  | ||||||
|             return result; |  | ||||||
|         } |  | ||||||
|         else |  | ||||||
|         { |  | ||||||
|             return Lists.newArrayList( options ); |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| @@ -0,0 +1,43 @@ | |||||||
|  | /* | ||||||
|  |  * This file is part of ComputerCraft - http://www.computercraft.info | ||||||
|  |  * Copyright Daniel Ratcliffe, 2011-2019. Do not distribute without permission. | ||||||
|  |  * Send enquiries to dratcliffe@gmail.com | ||||||
|  |  */ | ||||||
|  |  | ||||||
|  | package dan200.computercraft.shared.command; | ||||||
|  |  | ||||||
|  | import com.mojang.brigadier.exceptions.Dynamic2CommandExceptionType; | ||||||
|  | import com.mojang.brigadier.exceptions.DynamicCommandExceptionType; | ||||||
|  | import com.mojang.brigadier.exceptions.SimpleCommandExceptionType; | ||||||
|  | import net.minecraft.util.text.TextComponentTranslation; | ||||||
|  |  | ||||||
|  | public final class Exceptions | ||||||
|  | { | ||||||
|  |     public static final DynamicCommandExceptionType COMPUTER_ARG_NONE = translated1( "argument.computercraft.computer.no_matching" ); | ||||||
|  |     public static final Dynamic2CommandExceptionType COMPUTER_ARG_MANY = translated2( "argument.computercraft.computer.many_matching" ); | ||||||
|  |  | ||||||
|  |     public static final DynamicCommandExceptionType TRACKING_FIELD_ARG_NONE = translated1( "argument.computercraft.tacking_field.no_field" ); | ||||||
|  |  | ||||||
|  |     static final SimpleCommandExceptionType NOT_TRACKING_EXCEPTION = translated( "commands.computercraft.track.stop.not_enabled" ); | ||||||
|  |     static final SimpleCommandExceptionType NO_TIMINGS_EXCEPTION = translated( "commands.computercraft.track.dump.no_timings" ); | ||||||
|  |  | ||||||
|  |     static final SimpleCommandExceptionType TP_NOT_THERE = translated( "commands.computercraft.tp.not_there" ); | ||||||
|  |     static final SimpleCommandExceptionType TP_NOT_PLAYER = translated( "commands.computercraft.tp.not_player" ); | ||||||
|  |  | ||||||
|  |     public static final SimpleCommandExceptionType ARGUMENT_EXPECTED = translated( "argument.computercraft.argument_expected" ); | ||||||
|  |  | ||||||
|  |     private static SimpleCommandExceptionType translated( String key ) | ||||||
|  |     { | ||||||
|  |         return new SimpleCommandExceptionType( new TextComponentTranslation( key ) ); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     private static DynamicCommandExceptionType translated1( String key ) | ||||||
|  |     { | ||||||
|  |         return new DynamicCommandExceptionType( x -> new TextComponentTranslation( key, x ) ); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     private static Dynamic2CommandExceptionType translated2( String key ) | ||||||
|  |     { | ||||||
|  |         return new Dynamic2CommandExceptionType( ( x, y ) -> new TextComponentTranslation( key, x, y ) ); | ||||||
|  |     } | ||||||
|  | } | ||||||
| @@ -6,15 +6,17 @@ | |||||||
|  |  | ||||||
| package dan200.computercraft.shared.command; | package dan200.computercraft.shared.command; | ||||||
|  |  | ||||||
| import dan200.computercraft.shared.command.framework.CommandContext; | import net.minecraft.command.CommandSource; | ||||||
| import net.minecraft.command.ICommandSender; | import net.minecraft.entity.Entity; | ||||||
| import net.minecraft.entity.player.EntityPlayerMP; | import net.minecraft.entity.player.EntityPlayer; | ||||||
| import net.minecraft.server.MinecraftServer; | import net.minecraft.server.MinecraftServer; | ||||||
|  |  | ||||||
|  | import java.util.function.Predicate; | ||||||
|  |  | ||||||
| /** | /** | ||||||
|  * The level a user must be at in order to execute a command. |  * The level a user must be at in order to execute a command. | ||||||
|  */ |  */ | ||||||
| public enum UserLevel | public enum UserLevel implements Predicate<CommandSource> | ||||||
| { | { | ||||||
|     /** |     /** | ||||||
|      * Only can be used by the owner of the server: namely the server console or the player in SSP. |      * Only can be used by the owner of the server: namely the server console or the player in SSP. | ||||||
| @@ -51,20 +53,21 @@ public enum UserLevel | |||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     public boolean canExecute( CommandContext context ) |     @Override | ||||||
|  |     public boolean test( CommandSource source ) | ||||||
|     { |     { | ||||||
|         if( this == ANYONE ) return true; |         if( this == ANYONE ) return true; | ||||||
|  |  | ||||||
|         // We *always* allow level 0 stuff, even if the |         // We *always* allow level 0 stuff, even if the | ||||||
|         MinecraftServer server = context.getServer(); |         MinecraftServer server = source.getServer(); | ||||||
|         ICommandSender sender = context.getSender(); |         Entity sender = source.getEntity(); | ||||||
|  |  | ||||||
|         if( server.isSinglePlayer() && sender instanceof EntityPlayerMP && |         if( server.isSinglePlayer() && sender instanceof EntityPlayer && | ||||||
|             ((EntityPlayerMP) sender).getGameProfile().getName().equalsIgnoreCase( server.getServerOwner() ) ) |             ((EntityPlayer) sender).getGameProfile().getName().equalsIgnoreCase( server.getServerModName() ) ) | ||||||
|         { |         { | ||||||
|             if( this == OWNER || this == OWNER_OP ) return true; |             if( this == OWNER || this == OWNER_OP ) return true; | ||||||
|         } |         } | ||||||
|  |  | ||||||
|         return sender.canUseCommand( toLevel(), context.getRootCommand() ); |         return source.hasPermissionLevel( toLevel() ); | ||||||
|     } |     } | ||||||
| } | } | ||||||
|   | |||||||
| @@ -0,0 +1,41 @@ | |||||||
|  | /* | ||||||
|  |  * This file is part of ComputerCraft - http://www.computercraft.info | ||||||
|  |  * Copyright Daniel Ratcliffe, 2011-2019. Do not distribute without permission. | ||||||
|  |  * Send enquiries to dratcliffe@gmail.com | ||||||
|  |  */ | ||||||
|  |  | ||||||
|  | package dan200.computercraft.shared.command.arguments; | ||||||
|  |  | ||||||
|  | import com.mojang.brigadier.arguments.ArgumentType; | ||||||
|  | import dan200.computercraft.ComputerCraft; | ||||||
|  | import net.minecraft.command.arguments.ArgumentSerializer; | ||||||
|  | import net.minecraft.command.arguments.ArgumentTypes; | ||||||
|  | import net.minecraft.command.arguments.IArgumentSerializer; | ||||||
|  | import net.minecraft.util.ResourceLocation; | ||||||
|  |  | ||||||
|  | public final class ArgumentSerializers | ||||||
|  | { | ||||||
|  |     @SuppressWarnings( "unchecked" ) | ||||||
|  |     private static <T extends ArgumentType<?>> void registerUnsafe( ResourceLocation id, Class<T> type, IArgumentSerializer<?> serializer ) | ||||||
|  |     { | ||||||
|  |         ArgumentTypes.register( id, type, (IArgumentSerializer<T>) serializer ); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     private static <T extends ArgumentType<?>> void register( ResourceLocation id, Class<T> type, IArgumentSerializer<T> serializer ) | ||||||
|  |     { | ||||||
|  |         ArgumentTypes.register( id, type, serializer ); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     private static <T extends ArgumentType<?>> void register( ResourceLocation id, T instance ) | ||||||
|  |     { | ||||||
|  |         registerUnsafe( id, instance.getClass(), new ArgumentSerializer<>( () -> instance ) ); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     public static void register() | ||||||
|  |     { | ||||||
|  |         register( new ResourceLocation( ComputerCraft.MOD_ID, "tracking_field" ), TrackingFieldArgumentType.trackingField() ); | ||||||
|  |         register( new ResourceLocation( ComputerCraft.MOD_ID, "computer" ), ComputerArgumentType.oneComputer() ); | ||||||
|  |         register( new ResourceLocation( ComputerCraft.MOD_ID, "computers" ), ComputersArgumentType.class, new ComputersArgumentType.Serializer() ); | ||||||
|  |         registerUnsafe( new ResourceLocation( ComputerCraft.MOD_ID, "repeat" ), RepeatArgumentType.class, new RepeatArgumentType.Serializer() ); | ||||||
|  |     } | ||||||
|  | } | ||||||
| @@ -0,0 +1,75 @@ | |||||||
|  | /* | ||||||
|  |  * This file is part of ComputerCraft - http://www.computercraft.info | ||||||
|  |  * Copyright Daniel Ratcliffe, 2011-2019. Do not distribute without permission. | ||||||
|  |  * Send enquiries to dratcliffe@gmail.com | ||||||
|  |  */ | ||||||
|  |  | ||||||
|  | package dan200.computercraft.shared.command.arguments; | ||||||
|  |  | ||||||
|  | import com.mojang.brigadier.Message; | ||||||
|  | import com.mojang.brigadier.StringReader; | ||||||
|  | import com.mojang.brigadier.arguments.ArgumentType; | ||||||
|  | import com.mojang.brigadier.context.CommandContext; | ||||||
|  | import com.mojang.brigadier.exceptions.CommandSyntaxException; | ||||||
|  | import com.mojang.brigadier.exceptions.DynamicCommandExceptionType; | ||||||
|  | import com.mojang.brigadier.suggestion.Suggestions; | ||||||
|  | import com.mojang.brigadier.suggestion.SuggestionsBuilder; | ||||||
|  |  | ||||||
|  | import java.util.*; | ||||||
|  | import java.util.concurrent.CompletableFuture; | ||||||
|  | import java.util.function.Function; | ||||||
|  |  | ||||||
|  | public abstract class ChoiceArgumentType<T> implements ArgumentType<T> | ||||||
|  | { | ||||||
|  |     private final Iterable<T> choices; | ||||||
|  |     private final Function<T, String> name; | ||||||
|  |     private final Function<T, Message> tooltip; | ||||||
|  |     private final DynamicCommandExceptionType exception; | ||||||
|  |  | ||||||
|  |     protected ChoiceArgumentType( Iterable<T> choices, Function<T, String> name, Function<T, Message> tooltip, DynamicCommandExceptionType exception ) | ||||||
|  |     { | ||||||
|  |         this.choices = choices; | ||||||
|  |         this.name = name; | ||||||
|  |         this.tooltip = tooltip; | ||||||
|  |         this.exception = exception; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     @Override | ||||||
|  |     public T parse( StringReader reader ) throws CommandSyntaxException | ||||||
|  |     { | ||||||
|  |         int start = reader.getCursor(); | ||||||
|  |         String name = reader.readUnquotedString(); | ||||||
|  |  | ||||||
|  |         for( T choice : choices ) | ||||||
|  |         { | ||||||
|  |             String choiceName = this.name.apply( choice ); | ||||||
|  |             if( name.equals( choiceName ) ) return choice; | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         reader.setCursor( start ); | ||||||
|  |         throw exception.createWithContext( reader, name ); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     @Override | ||||||
|  |     public <S> CompletableFuture<Suggestions> listSuggestions( CommandContext<S> context, SuggestionsBuilder builder ) | ||||||
|  |     { | ||||||
|  |         String remaining = builder.getRemaining().toLowerCase( Locale.ROOT ); | ||||||
|  |         for( T choice : choices ) | ||||||
|  |         { | ||||||
|  |             String name = this.name.apply( choice ); | ||||||
|  |             if( !name.toLowerCase( Locale.ROOT ).startsWith( remaining ) ) continue; | ||||||
|  |             builder.suggest( name, tooltip.apply( choice ) ); | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         return builder.buildFuture(); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     @Override | ||||||
|  |     public Collection<String> getExamples() | ||||||
|  |     { | ||||||
|  |         List<String> items = choices instanceof Collection<?> ? new ArrayList<>( ((Collection<T>) choices).size() ) : new ArrayList<>(); | ||||||
|  |         for( T choice : choices ) items.add( name.apply( choice ) ); | ||||||
|  |         items.sort( Comparator.naturalOrder() ); | ||||||
|  |         return items; | ||||||
|  |     } | ||||||
|  | } | ||||||
| @@ -0,0 +1,94 @@ | |||||||
|  | /* | ||||||
|  |  * This file is part of ComputerCraft - http://www.computercraft.info | ||||||
|  |  * Copyright Daniel Ratcliffe, 2011-2019. Do not distribute without permission. | ||||||
|  |  * Send enquiries to dratcliffe@gmail.com | ||||||
|  |  */ | ||||||
|  |  | ||||||
|  | package dan200.computercraft.shared.command.arguments; | ||||||
|  |  | ||||||
|  | import com.mojang.brigadier.StringReader; | ||||||
|  | import com.mojang.brigadier.arguments.ArgumentType; | ||||||
|  | import com.mojang.brigadier.context.CommandContext; | ||||||
|  | import com.mojang.brigadier.exceptions.CommandSyntaxException; | ||||||
|  | import com.mojang.brigadier.suggestion.Suggestions; | ||||||
|  | import com.mojang.brigadier.suggestion.SuggestionsBuilder; | ||||||
|  | import dan200.computercraft.shared.command.arguments.ComputersArgumentType.ComputersSupplier; | ||||||
|  | import dan200.computercraft.shared.computer.core.ServerComputer; | ||||||
|  | import net.minecraft.command.CommandSource; | ||||||
|  |  | ||||||
|  | import java.util.Collection; | ||||||
|  | import java.util.concurrent.CompletableFuture; | ||||||
|  |  | ||||||
|  | import static dan200.computercraft.shared.command.Exceptions.COMPUTER_ARG_MANY; | ||||||
|  |  | ||||||
|  | public final class ComputerArgumentType implements ArgumentType<ComputerArgumentType.ComputerSupplier> | ||||||
|  | { | ||||||
|  |     private static final ComputerArgumentType INSTANCE = new ComputerArgumentType(); | ||||||
|  |  | ||||||
|  |     public static ComputerArgumentType oneComputer() | ||||||
|  |     { | ||||||
|  |         return INSTANCE; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     public static ServerComputer getComputerArgument( CommandContext<CommandSource> context, String name ) throws CommandSyntaxException | ||||||
|  |     { | ||||||
|  |         return context.getArgument( name, ComputerSupplier.class ).unwrap( context.getSource() ); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     private ComputerArgumentType() | ||||||
|  |     { | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     @Override | ||||||
|  |     public ComputerSupplier parse( StringReader reader ) throws CommandSyntaxException | ||||||
|  |     { | ||||||
|  |         int start = reader.getCursor(); | ||||||
|  |         ComputersSupplier supplier = ComputersArgumentType.someComputers().parse( reader ); | ||||||
|  |         String selector = reader.getString().substring( start, reader.getCursor() ); | ||||||
|  |  | ||||||
|  |         return s -> { | ||||||
|  |             Collection<ServerComputer> computers = supplier.unwrap( s ); | ||||||
|  |  | ||||||
|  |             if( computers.size() == 1 ) return computers.iterator().next(); | ||||||
|  |  | ||||||
|  |             StringBuilder builder = new StringBuilder(); | ||||||
|  |             boolean first = true; | ||||||
|  |             for( ServerComputer computer : computers ) | ||||||
|  |             { | ||||||
|  |                 if( first ) | ||||||
|  |                 { | ||||||
|  |                     first = false; | ||||||
|  |                 } | ||||||
|  |                 else | ||||||
|  |                 { | ||||||
|  |                     builder.append( ", " ); | ||||||
|  |                 } | ||||||
|  |  | ||||||
|  |                 builder.append( computer.getInstanceID() ); | ||||||
|  |             } | ||||||
|  |  | ||||||
|  |  | ||||||
|  |             // We have an incorrect number of computers: reset and throw an error | ||||||
|  |             reader.setCursor( start ); | ||||||
|  |             throw COMPUTER_ARG_MANY.createWithContext( reader, selector, builder.toString() ); | ||||||
|  |         }; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     @Override | ||||||
|  |     public <S> CompletableFuture<Suggestions> listSuggestions( CommandContext<S> context, SuggestionsBuilder builder ) | ||||||
|  |     { | ||||||
|  |         return ComputersArgumentType.someComputers().listSuggestions( context, builder ); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     @Override | ||||||
|  |     public Collection<String> getExamples() | ||||||
|  |     { | ||||||
|  |         return ComputersArgumentType.someComputers().getExamples(); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     @FunctionalInterface | ||||||
|  |     public interface ComputerSupplier | ||||||
|  |     { | ||||||
|  |         ServerComputer unwrap( CommandSource source ) throws CommandSyntaxException; | ||||||
|  |     } | ||||||
|  | } | ||||||
| @@ -0,0 +1,210 @@ | |||||||
|  | /* | ||||||
|  |  * This file is part of ComputerCraft - http://www.computercraft.info | ||||||
|  |  * Copyright Daniel Ratcliffe, 2011-2019. Do not distribute without permission. | ||||||
|  |  * Send enquiries to dratcliffe@gmail.com | ||||||
|  |  */ | ||||||
|  |  | ||||||
|  | package dan200.computercraft.shared.command.arguments; | ||||||
|  |  | ||||||
|  | import com.google.gson.JsonObject; | ||||||
|  | import com.mojang.brigadier.StringReader; | ||||||
|  | import com.mojang.brigadier.arguments.ArgumentType; | ||||||
|  | import com.mojang.brigadier.context.CommandContext; | ||||||
|  | import com.mojang.brigadier.exceptions.CommandSyntaxException; | ||||||
|  | import com.mojang.brigadier.suggestion.Suggestions; | ||||||
|  | import com.mojang.brigadier.suggestion.SuggestionsBuilder; | ||||||
|  | import dan200.computercraft.ComputerCraft; | ||||||
|  | import dan200.computercraft.shared.computer.core.ComputerFamily; | ||||||
|  | import dan200.computercraft.shared.computer.core.ServerComputer; | ||||||
|  | import net.minecraft.command.CommandSource; | ||||||
|  | import net.minecraft.command.arguments.IArgumentSerializer; | ||||||
|  | import net.minecraft.network.PacketBuffer; | ||||||
|  |  | ||||||
|  | import javax.annotation.Nonnull; | ||||||
|  | import java.util.*; | ||||||
|  | import java.util.concurrent.CompletableFuture; | ||||||
|  | import java.util.function.Function; | ||||||
|  | import java.util.function.Predicate; | ||||||
|  | import java.util.stream.Collectors; | ||||||
|  |  | ||||||
|  | import static dan200.computercraft.shared.command.CommandUtils.suggest; | ||||||
|  | import static dan200.computercraft.shared.command.CommandUtils.suggestOnServer; | ||||||
|  | import static dan200.computercraft.shared.command.Exceptions.COMPUTER_ARG_NONE; | ||||||
|  |  | ||||||
|  | public final class ComputersArgumentType implements ArgumentType<ComputersArgumentType.ComputersSupplier> | ||||||
|  | { | ||||||
|  |     private static final ComputersArgumentType MANY = new ComputersArgumentType( false ); | ||||||
|  |     private static final ComputersArgumentType SOME = new ComputersArgumentType( true ); | ||||||
|  |  | ||||||
|  |     private static final List<String> EXAMPLES = Arrays.asList( | ||||||
|  |         "0", "#0", "@Label", "~Advanced" | ||||||
|  |     ); | ||||||
|  |  | ||||||
|  |     public static ComputersArgumentType manyComputers() | ||||||
|  |     { | ||||||
|  |         return MANY; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     public static ComputersArgumentType someComputers() | ||||||
|  |     { | ||||||
|  |         return SOME; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     public static Collection<ServerComputer> getComputersArgument( CommandContext<CommandSource> context, String name ) throws CommandSyntaxException | ||||||
|  |     { | ||||||
|  |         return context.getArgument( name, ComputersSupplier.class ).unwrap( context.getSource() ); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     private final boolean requireSome; | ||||||
|  |  | ||||||
|  |     private ComputersArgumentType( boolean requireSome ) | ||||||
|  |     { | ||||||
|  |         this.requireSome = requireSome; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     @Override | ||||||
|  |     public ComputersSupplier parse( StringReader reader ) throws CommandSyntaxException | ||||||
|  |     { | ||||||
|  |         int start = reader.getCursor(); | ||||||
|  |         char kind = reader.peek(); | ||||||
|  |         ComputersSupplier computers; | ||||||
|  |         if( kind == '@' ) | ||||||
|  |         { | ||||||
|  |             reader.skip(); | ||||||
|  |             String label = reader.readUnquotedString(); | ||||||
|  |             computers = getComputers( x -> Objects.equals( label, x.getLabel() ) ); | ||||||
|  |         } | ||||||
|  |         else if( kind == '~' ) | ||||||
|  |         { | ||||||
|  |             reader.skip(); | ||||||
|  |             String family = reader.readUnquotedString(); | ||||||
|  |             computers = getComputers( x -> x.getFamily().name().equalsIgnoreCase( family ) ); | ||||||
|  |         } | ||||||
|  |         else if( kind == '#' ) | ||||||
|  |         { | ||||||
|  |             reader.skip(); | ||||||
|  |             int id = reader.readInt(); | ||||||
|  |             computers = getComputers( x -> x.getID() == id ); | ||||||
|  |         } | ||||||
|  |         else | ||||||
|  |         { | ||||||
|  |             int instance = reader.readInt(); | ||||||
|  |             computers = s -> { | ||||||
|  |                 ServerComputer computer = ComputerCraft.serverComputerRegistry.get( instance ); | ||||||
|  |                 return computer == null ? Collections.emptyList() : Collections.singletonList( computer ); | ||||||
|  |             }; | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         if( requireSome ) | ||||||
|  |         { | ||||||
|  |             String selector = reader.getString().substring( start, reader.getCursor() ); | ||||||
|  |             return source -> { | ||||||
|  |                 Collection<ServerComputer> matched = computers.unwrap( source ); | ||||||
|  |                 if( matched.isEmpty() ) throw COMPUTER_ARG_NONE.create( selector ); | ||||||
|  |                 return matched; | ||||||
|  |             }; | ||||||
|  |         } | ||||||
|  |         else | ||||||
|  |         { | ||||||
|  |             return computers; | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     @Override | ||||||
|  |     public <S> CompletableFuture<Suggestions> listSuggestions( CommandContext<S> context, SuggestionsBuilder builder ) | ||||||
|  |     { | ||||||
|  |         String remaining = builder.getRemaining(); | ||||||
|  |  | ||||||
|  |         // We can run this one on the client, for obvious reasons. | ||||||
|  |         if( remaining.startsWith( "~" ) ) | ||||||
|  |         { | ||||||
|  |             return suggest( builder, ComputerFamily.values(), x -> "~" + x.name() ); | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         // Verify we've a command source and we're running on the server | ||||||
|  |         return suggestOnServer( context, builder, s -> { | ||||||
|  |             if( remaining.startsWith( "@" ) ) | ||||||
|  |             { | ||||||
|  |                 suggestComputers( builder, remaining, x -> { | ||||||
|  |                     String label = x.getLabel(); | ||||||
|  |                     return label == null ? null : "@" + label; | ||||||
|  |                 } ); | ||||||
|  |             } | ||||||
|  |             else if( remaining.startsWith( "#" ) ) | ||||||
|  |             { | ||||||
|  |                 suggestComputers( builder, remaining, c -> "#" + c.getID() ); | ||||||
|  |             } | ||||||
|  |             else | ||||||
|  |             { | ||||||
|  |                 suggestComputers( builder, remaining, c -> Integer.toString( c.getInstanceID() ) ); | ||||||
|  |             } | ||||||
|  |  | ||||||
|  |             return builder.buildFuture(); | ||||||
|  |         } ); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     @Override | ||||||
|  |     public Collection<String> getExamples() | ||||||
|  |     { | ||||||
|  |         return EXAMPLES; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     private static void suggestComputers( SuggestionsBuilder builder, String remaining, Function<ServerComputer, String> renderer ) | ||||||
|  |     { | ||||||
|  |         remaining = remaining.toLowerCase( Locale.ROOT ); | ||||||
|  |         for( ServerComputer computer : ComputerCraft.serverComputerRegistry.getComputers() ) | ||||||
|  |         { | ||||||
|  |             String converted = renderer.apply( computer ); | ||||||
|  |             if( converted != null && converted.toLowerCase( Locale.ROOT ).startsWith( remaining ) ) | ||||||
|  |             { | ||||||
|  |                 builder.suggest( converted ); | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     private static ComputersSupplier getComputers( Predicate<ServerComputer> predicate ) | ||||||
|  |     { | ||||||
|  |         return s -> Collections.unmodifiableList( ComputerCraft.serverComputerRegistry | ||||||
|  |             .getComputers() | ||||||
|  |             .stream() | ||||||
|  |             .filter( predicate ) | ||||||
|  |             .collect( Collectors.toList() ) | ||||||
|  |         ); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     public static class Serializer implements IArgumentSerializer<ComputersArgumentType> | ||||||
|  |     { | ||||||
|  |  | ||||||
|  |         @Override | ||||||
|  |         public void write( @Nonnull ComputersArgumentType arg, @Nonnull PacketBuffer buf ) | ||||||
|  |         { | ||||||
|  |             buf.writeBoolean( arg.requireSome ); | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         @Nonnull | ||||||
|  |         @Override | ||||||
|  |         public ComputersArgumentType read( @Nonnull PacketBuffer buf ) | ||||||
|  |         { | ||||||
|  |             return buf.readBoolean() ? SOME : MANY; | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         @Override | ||||||
|  |         public void write( @Nonnull ComputersArgumentType arg, @Nonnull JsonObject json ) | ||||||
|  |         { | ||||||
|  |             json.addProperty( "requireSome", arg.requireSome ); | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     @FunctionalInterface | ||||||
|  |     public interface ComputersSupplier | ||||||
|  |     { | ||||||
|  |         Collection<ServerComputer> unwrap( CommandSource source ) throws CommandSyntaxException; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     public static Set<ServerComputer> unwrap( CommandSource source, Collection<ComputersSupplier> suppliers ) throws CommandSyntaxException | ||||||
|  |     { | ||||||
|  |         Set<ServerComputer> computers = new HashSet<>(); | ||||||
|  |         for( ComputersSupplier supplier : suppliers ) computers.addAll( supplier.unwrap( source ) ); | ||||||
|  |         return computers; | ||||||
|  |     } | ||||||
|  | } | ||||||
| @@ -0,0 +1,166 @@ | |||||||
|  | /* | ||||||
|  |  * This file is part of ComputerCraft - http://www.computercraft.info | ||||||
|  |  * Copyright Daniel Ratcliffe, 2011-2019. Do not distribute without permission. | ||||||
|  |  * Send enquiries to dratcliffe@gmail.com | ||||||
|  |  */ | ||||||
|  |  | ||||||
|  | package dan200.computercraft.shared.command.arguments; | ||||||
|  |  | ||||||
|  | import com.google.gson.JsonObject; | ||||||
|  | import com.mojang.brigadier.Message; | ||||||
|  | import com.mojang.brigadier.StringReader; | ||||||
|  | import com.mojang.brigadier.arguments.ArgumentType; | ||||||
|  | import com.mojang.brigadier.context.CommandContext; | ||||||
|  | import com.mojang.brigadier.exceptions.CommandSyntaxException; | ||||||
|  | import com.mojang.brigadier.exceptions.SimpleCommandExceptionType; | ||||||
|  | import com.mojang.brigadier.suggestion.Suggestions; | ||||||
|  | import com.mojang.brigadier.suggestion.SuggestionsBuilder; | ||||||
|  | import net.minecraft.command.arguments.ArgumentTypes; | ||||||
|  | import net.minecraft.command.arguments.IArgumentSerializer; | ||||||
|  | import net.minecraft.network.PacketBuffer; | ||||||
|  | import net.minecraft.util.text.ITextComponent; | ||||||
|  | import net.minecraft.util.text.TextComponentString; | ||||||
|  |  | ||||||
|  | import javax.annotation.Nonnull; | ||||||
|  | import java.util.ArrayList; | ||||||
|  | import java.util.Collection; | ||||||
|  | import java.util.Collections; | ||||||
|  | import java.util.List; | ||||||
|  | import java.util.concurrent.CompletableFuture; | ||||||
|  | import java.util.function.BiConsumer; | ||||||
|  |  | ||||||
|  | /** | ||||||
|  |  * Reads one argument multiple times. | ||||||
|  |  * | ||||||
|  |  * Note that this must be the last element in an argument chain: in order to improve the quality of error messages, | ||||||
|  |  * we will always try to consume another argument while there is input remaining. | ||||||
|  |  * | ||||||
|  |  * One problem with how parsers function, is that they must consume some input: and thus we | ||||||
|  |  * | ||||||
|  |  * @param <T> The type of each value returned | ||||||
|  |  * @param <U> The type of the inner parser. This will normally be a {@link List} or {@code T}. | ||||||
|  |  */ | ||||||
|  | public final class RepeatArgumentType<T, U> implements ArgumentType<List<T>> | ||||||
|  | { | ||||||
|  |     private final ArgumentType<U> child; | ||||||
|  |     private final BiConsumer<List<T>, U> appender; | ||||||
|  |     private final boolean flatten; | ||||||
|  |     private final SimpleCommandExceptionType some; | ||||||
|  |  | ||||||
|  |     private RepeatArgumentType( ArgumentType<U> child, BiConsumer<List<T>, U> appender, boolean flatten, SimpleCommandExceptionType some ) | ||||||
|  |     { | ||||||
|  |         this.child = child; | ||||||
|  |         this.appender = appender; | ||||||
|  |         this.flatten = flatten; | ||||||
|  |         this.some = some; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     public static <T> RepeatArgumentType<T, T> some( ArgumentType<T> appender, SimpleCommandExceptionType missing ) | ||||||
|  |     { | ||||||
|  |         return new RepeatArgumentType<>( appender, List::add, true, missing ); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     public static <T> RepeatArgumentType<T, List<T>> someFlat( ArgumentType<List<T>> appender, SimpleCommandExceptionType missing ) | ||||||
|  |     { | ||||||
|  |         return new RepeatArgumentType<>( appender, List::addAll, true, missing ); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     @Override | ||||||
|  |     public List<T> parse( StringReader reader ) throws CommandSyntaxException | ||||||
|  |     { | ||||||
|  |         boolean hadSome = false; | ||||||
|  |         List<T> out = new ArrayList<>(); | ||||||
|  |         while( true ) | ||||||
|  |         { | ||||||
|  |             reader.skipWhitespace(); | ||||||
|  |             if( !reader.canRead() ) break; | ||||||
|  |  | ||||||
|  |             int startParse = reader.getCursor(); | ||||||
|  |             appender.accept( out, child.parse( reader ) ); | ||||||
|  |             hadSome = true; | ||||||
|  |  | ||||||
|  |             if( reader.getCursor() == startParse ) | ||||||
|  |             { | ||||||
|  |                 throw new IllegalStateException( child + " did not consume any input on " + reader.getRemaining() ); | ||||||
|  |             } | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         // Note that each child may return an empty list, we just require that some actual input | ||||||
|  |         // was consumed. | ||||||
|  |         // We should probably review that this is sensible in the future. | ||||||
|  |         if( !hadSome ) throw some.createWithContext( reader ); | ||||||
|  |  | ||||||
|  |         return Collections.unmodifiableList( out ); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     @Override | ||||||
|  |     public <S> CompletableFuture<Suggestions> listSuggestions( CommandContext<S> context, SuggestionsBuilder builder ) | ||||||
|  |     { | ||||||
|  |         StringReader reader = new StringReader( builder.getInput() ); | ||||||
|  |         reader.setCursor( builder.getStart() ); | ||||||
|  |         int previous = reader.getCursor(); | ||||||
|  |         while( reader.canRead() ) | ||||||
|  |         { | ||||||
|  |             try | ||||||
|  |             { | ||||||
|  |                 child.parse( reader ); | ||||||
|  |             } | ||||||
|  |             catch( CommandSyntaxException e ) | ||||||
|  |             { | ||||||
|  |                 break; | ||||||
|  |             } | ||||||
|  |  | ||||||
|  |             int cursor = reader.getCursor(); | ||||||
|  |             reader.skipWhitespace(); | ||||||
|  |             if( cursor == reader.getCursor() ) break; | ||||||
|  |             previous = reader.getCursor(); | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         reader.setCursor( previous ); | ||||||
|  |         return child.listSuggestions( context, builder.createOffset( previous ) ); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     @Override | ||||||
|  |     public Collection<String> getExamples() | ||||||
|  |     { | ||||||
|  |         return child.getExamples(); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     public static class Serializer implements IArgumentSerializer<RepeatArgumentType<?, ?>> | ||||||
|  |     { | ||||||
|  |         @Override | ||||||
|  |         public void write( @Nonnull RepeatArgumentType<?, ?> arg, @Nonnull PacketBuffer buf ) | ||||||
|  |         { | ||||||
|  |             buf.writeBoolean( arg.flatten ); | ||||||
|  |             ArgumentTypes.serialize( buf, arg.child ); | ||||||
|  |             buf.writeTextComponent( getMessage( arg ) ); | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         @Nonnull | ||||||
|  |         @Override | ||||||
|  |         @SuppressWarnings( { "unchecked", "rawtypes" } ) | ||||||
|  |         public RepeatArgumentType<?, ?> read( @Nonnull PacketBuffer buf ) | ||||||
|  |         { | ||||||
|  |             boolean isList = buf.readBoolean(); | ||||||
|  |             ArgumentType<?> child = ArgumentTypes.deserialize( buf ); | ||||||
|  |             ITextComponent message = buf.readTextComponent(); | ||||||
|  |             BiConsumer<List<Object>, ?> appender = isList ? ( list, x ) -> list.addAll( (Collection) x ) : List::add; | ||||||
|  |             return new RepeatArgumentType( child, appender, isList, new SimpleCommandExceptionType( message ) ); | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         @Override | ||||||
|  |         public void write( @Nonnull RepeatArgumentType<?, ?> arg, @Nonnull JsonObject json ) | ||||||
|  |         { | ||||||
|  |             json.addProperty( "flatten", arg.flatten ); | ||||||
|  |             json.addProperty( "child", "<<cannot serialize>>" ); // TODO: Potentially serialize this using reflection. | ||||||
|  |             json.addProperty( "error", ITextComponent.Serializer.toJson( getMessage( arg ) ) ); | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         private static ITextComponent getMessage( RepeatArgumentType<?, ?> arg ) | ||||||
|  |         { | ||||||
|  |             Message message = arg.some.create().getRawMessage(); | ||||||
|  |             if( message instanceof ITextComponent ) return (ITextComponent) message; | ||||||
|  |             return new TextComponentString( message.getString() ); | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | } | ||||||
| @@ -0,0 +1,27 @@ | |||||||
|  | /* | ||||||
|  |  * This file is part of ComputerCraft - http://www.computercraft.info | ||||||
|  |  * Copyright Daniel Ratcliffe, 2011-2019. Do not distribute without permission. | ||||||
|  |  * Send enquiries to dratcliffe@gmail.com | ||||||
|  |  */ | ||||||
|  |  | ||||||
|  | package dan200.computercraft.shared.command.arguments; | ||||||
|  |  | ||||||
|  | import dan200.computercraft.core.tracking.TrackingField; | ||||||
|  | import dan200.computercraft.shared.command.Exceptions; | ||||||
|  |  | ||||||
|  | import static dan200.computercraft.shared.command.text.ChatHelpers.translate; | ||||||
|  |  | ||||||
|  | public final class TrackingFieldArgumentType extends ChoiceArgumentType<TrackingField> | ||||||
|  | { | ||||||
|  |     private static final TrackingFieldArgumentType INSTANCE = new TrackingFieldArgumentType(); | ||||||
|  |  | ||||||
|  |     private TrackingFieldArgumentType() | ||||||
|  |     { | ||||||
|  |         super( TrackingField.fields().values(), TrackingField::id, x -> translate( x.translationKey() ), Exceptions.TRACKING_FIELD_ARG_NONE ); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     public static TrackingFieldArgumentType trackingField() | ||||||
|  |     { | ||||||
|  |         return INSTANCE; | ||||||
|  |     } | ||||||
|  | } | ||||||
| @@ -0,0 +1,20 @@ | |||||||
|  | /* | ||||||
|  |  * This file is part of ComputerCraft - http://www.computercraft.info | ||||||
|  |  * Copyright Daniel Ratcliffe, 2011-2019. Do not distribute without permission. | ||||||
|  |  * Send enquiries to dratcliffe@gmail.com | ||||||
|  |  */ | ||||||
|  |  | ||||||
|  | package dan200.computercraft.shared.command.builder; | ||||||
|  |  | ||||||
|  | import com.mojang.brigadier.Command; | ||||||
|  | import com.mojang.brigadier.context.CommandContext; | ||||||
|  | import com.mojang.brigadier.exceptions.CommandSyntaxException; | ||||||
|  |  | ||||||
|  | /** | ||||||
|  |  * A {@link Command} which accepts an argument | ||||||
|  |  */ | ||||||
|  | @FunctionalInterface | ||||||
|  | public interface ArgCommand<S, T> | ||||||
|  | { | ||||||
|  |     int run( CommandContext<S> ctx, T arg ) throws CommandSyntaxException; | ||||||
|  | } | ||||||
| @@ -0,0 +1,126 @@ | |||||||
|  | /* | ||||||
|  |  * This file is part of ComputerCraft - http://www.computercraft.info | ||||||
|  |  * Copyright Daniel Ratcliffe, 2011-2019. Do not distribute without permission. | ||||||
|  |  * Send enquiries to dratcliffe@gmail.com | ||||||
|  |  */ | ||||||
|  |  | ||||||
|  | package dan200.computercraft.shared.command.builder; | ||||||
|  |  | ||||||
|  | import com.mojang.brigadier.Command; | ||||||
|  | import com.mojang.brigadier.arguments.ArgumentType; | ||||||
|  | import com.mojang.brigadier.builder.ArgumentBuilder; | ||||||
|  | import com.mojang.brigadier.builder.RequiredArgumentBuilder; | ||||||
|  | import com.mojang.brigadier.context.CommandContext; | ||||||
|  | import com.mojang.brigadier.tree.CommandNode; | ||||||
|  | import dan200.computercraft.shared.command.arguments.RepeatArgumentType; | ||||||
|  | import net.minecraft.command.CommandSource; | ||||||
|  |  | ||||||
|  | import java.util.ArrayList; | ||||||
|  | import java.util.Collections; | ||||||
|  | import java.util.List; | ||||||
|  | import java.util.function.Predicate; | ||||||
|  | import java.util.function.Supplier; | ||||||
|  |  | ||||||
|  | import static dan200.computercraft.shared.command.Exceptions.ARGUMENT_EXPECTED; | ||||||
|  | import static dan200.computercraft.shared.command.builder.HelpingArgumentBuilder.literal; | ||||||
|  |  | ||||||
|  | /** | ||||||
|  |  * An alternative way of building command nodes, so one does not have to nest | ||||||
|  |  * {@link ArgumentBuilder#then(CommandNode)}s. | ||||||
|  |  */ | ||||||
|  | public class CommandBuilder<S> implements CommandNodeBuilder<S, Command<S>> | ||||||
|  | { | ||||||
|  |     private List<ArgumentBuilder<S, ?>> args = new ArrayList<>(); | ||||||
|  |     private Predicate<S> requires; | ||||||
|  |  | ||||||
|  |     public static CommandBuilder<CommandSource> args() | ||||||
|  |     { | ||||||
|  |         return new CommandBuilder<>(); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     public static CommandBuilder<CommandSource> command( String literal ) | ||||||
|  |     { | ||||||
|  |         CommandBuilder<CommandSource> builder = new CommandBuilder<>(); | ||||||
|  |         builder.args.add( literal( literal ) ); | ||||||
|  |         return builder; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     public CommandBuilder<S> requires( Predicate<S> predicate ) | ||||||
|  |     { | ||||||
|  |         requires = requires == null ? predicate : requires.and( predicate ); | ||||||
|  |         return this; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     public CommandBuilder<S> arg( String name, ArgumentType<?> type ) | ||||||
|  |     { | ||||||
|  |         args.add( RequiredArgumentBuilder.argument( name, type ) ); | ||||||
|  |         return this; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     public <T> CommandNodeBuilder<S, ArgCommand<S, List<T>>> argManyValue( String name, ArgumentType<T> type, List<T> empty ) | ||||||
|  |     { | ||||||
|  |         return argMany( name, type, () -> empty ); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     public <T> CommandNodeBuilder<S, ArgCommand<S, List<T>>> argManyValue( String name, ArgumentType<T> type, T defaultValue ) | ||||||
|  |     { | ||||||
|  |         return argManyValue( name, type, Collections.singletonList( defaultValue ) ); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     public <T> CommandNodeBuilder<S, ArgCommand<S, List<T>>> argMany( String name, ArgumentType<T> type, Supplier<List<T>> empty ) | ||||||
|  |     { | ||||||
|  |         return argMany( name, RepeatArgumentType.some( type, ARGUMENT_EXPECTED ), empty ); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     public <T> CommandNodeBuilder<S, ArgCommand<S, List<T>>> argManyFlatten( String name, ArgumentType<List<T>> type, Supplier<List<T>> empty ) | ||||||
|  |     { | ||||||
|  |         return argMany( name, RepeatArgumentType.someFlat( type, ARGUMENT_EXPECTED ), empty ); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     private <T, U> CommandNodeBuilder<S, ArgCommand<S, List<T>>> argMany( String name, RepeatArgumentType<T, ?> type, Supplier<List<T>> empty ) | ||||||
|  |     { | ||||||
|  |         if( args.isEmpty() ) throw new IllegalStateException( "Cannot have empty arg chain builder" ); | ||||||
|  |  | ||||||
|  |         return command -> { | ||||||
|  |             // The node for no arguments | ||||||
|  |             ArgumentBuilder<S, ?> tail = tail( ctx -> command.run( ctx, empty.get() ) ); | ||||||
|  |  | ||||||
|  |             // The node for one or more arguments | ||||||
|  |             ArgumentBuilder<S, ?> moreArg = RequiredArgumentBuilder | ||||||
|  |                 .<S, List<T>>argument( name, type ) | ||||||
|  |                 .executes( ctx -> command.run( ctx, getList( ctx, name ) ) ); | ||||||
|  |  | ||||||
|  |             // Chain all of them together! | ||||||
|  |             tail.then( moreArg ); | ||||||
|  |             return link( tail ); | ||||||
|  |         }; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     @SuppressWarnings( "unchecked" ) | ||||||
|  |     private static <T> List<T> getList( CommandContext<?> context, String name ) | ||||||
|  |     { | ||||||
|  |         return (List<T>) context.getArgument( name, List.class ); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     @Override | ||||||
|  |     public CommandNode<S> executes( Command<S> command ) | ||||||
|  |     { | ||||||
|  |         if( args.isEmpty() ) throw new IllegalStateException( "Cannot have empty arg chain builder" ); | ||||||
|  |  | ||||||
|  |         return link( tail( command ) ); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     private ArgumentBuilder<S, ?> tail( Command<S> command ) | ||||||
|  |     { | ||||||
|  |         ArgumentBuilder<S, ?> defaultTail = args.get( args.size() - 1 ); | ||||||
|  |         defaultTail.executes( command ); | ||||||
|  |         if( requires != null ) defaultTail.requires( requires ); | ||||||
|  |         return defaultTail; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     private CommandNode<S> link( ArgumentBuilder<S, ?> tail ) | ||||||
|  |     { | ||||||
|  |         for( int i = args.size() - 2; i >= 0; i-- ) tail = args.get( i ).then( tail ); | ||||||
|  |         return tail.build(); | ||||||
|  |     } | ||||||
|  | } | ||||||
| @@ -0,0 +1,24 @@ | |||||||
|  | /* | ||||||
|  |  * This file is part of ComputerCraft - http://www.computercraft.info | ||||||
|  |  * Copyright Daniel Ratcliffe, 2011-2019. Do not distribute without permission. | ||||||
|  |  * Send enquiries to dratcliffe@gmail.com | ||||||
|  |  */ | ||||||
|  |  | ||||||
|  | package dan200.computercraft.shared.command.builder; | ||||||
|  |  | ||||||
|  | import com.mojang.brigadier.tree.CommandNode; | ||||||
|  |  | ||||||
|  | /** | ||||||
|  |  * A builder which generates a {@link CommandNode} from the provided action. | ||||||
|  |  */ | ||||||
|  | @FunctionalInterface | ||||||
|  | public interface CommandNodeBuilder<S, T> | ||||||
|  | { | ||||||
|  |     /** | ||||||
|  |      * Generate a command node which executes this command. | ||||||
|  |      * | ||||||
|  |      * @param command The command to run | ||||||
|  |      * @return The constructed node. | ||||||
|  |      */ | ||||||
|  |     CommandNode<S> executes( T command ); | ||||||
|  | } | ||||||
| @@ -0,0 +1,205 @@ | |||||||
|  | /* | ||||||
|  |  * This file is part of ComputerCraft - http://www.computercraft.info | ||||||
|  |  * Copyright Daniel Ratcliffe, 2011-2019. Do not distribute without permission. | ||||||
|  |  * Send enquiries to dratcliffe@gmail.com | ||||||
|  |  */ | ||||||
|  |  | ||||||
|  | package dan200.computercraft.shared.command.builder; | ||||||
|  |  | ||||||
|  | import com.mojang.brigadier.Command; | ||||||
|  | import com.mojang.brigadier.CommandDispatcher; | ||||||
|  | import com.mojang.brigadier.builder.ArgumentBuilder; | ||||||
|  | import com.mojang.brigadier.builder.LiteralArgumentBuilder; | ||||||
|  | import com.mojang.brigadier.context.CommandContext; | ||||||
|  | import com.mojang.brigadier.tree.CommandNode; | ||||||
|  | import com.mojang.brigadier.tree.LiteralCommandNode; | ||||||
|  | import net.minecraft.command.CommandSource; | ||||||
|  | import net.minecraft.util.text.ITextComponent; | ||||||
|  | import net.minecraft.util.text.TextComponentString; | ||||||
|  | import net.minecraft.util.text.TextFormatting; | ||||||
|  | import net.minecraft.util.text.event.ClickEvent; | ||||||
|  |  | ||||||
|  | import javax.annotation.Nonnull; | ||||||
|  | import java.util.ArrayList; | ||||||
|  | import java.util.Collection; | ||||||
|  |  | ||||||
|  | import static dan200.computercraft.shared.command.text.ChatHelpers.coloured; | ||||||
|  | import static dan200.computercraft.shared.command.text.ChatHelpers.translate; | ||||||
|  |  | ||||||
|  | /** | ||||||
|  |  * An alternative to {@link LiteralArgumentBuilder} which also provides a {@code /... help} command, and defaults | ||||||
|  |  * to that command when no arguments are given. | ||||||
|  |  */ | ||||||
|  | public final class HelpingArgumentBuilder extends LiteralArgumentBuilder<CommandSource> | ||||||
|  | { | ||||||
|  |     private final Collection<HelpingArgumentBuilder> children = new ArrayList<>(); | ||||||
|  |  | ||||||
|  |     private HelpingArgumentBuilder( String literal ) | ||||||
|  |     { | ||||||
|  |         super( literal ); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     public static HelpingArgumentBuilder choice( String literal ) | ||||||
|  |     { | ||||||
|  |         return new HelpingArgumentBuilder( literal ); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     @Override | ||||||
|  |     public LiteralArgumentBuilder<CommandSource> executes( final Command<CommandSource> command ) | ||||||
|  |     { | ||||||
|  |         throw new IllegalStateException( "Cannot use executes on a HelpingArgumentBuilder" ); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     @Override | ||||||
|  |     public LiteralArgumentBuilder<CommandSource> then( final ArgumentBuilder<CommandSource, ?> argument ) | ||||||
|  |     { | ||||||
|  |         if( getRedirect() != null ) throw new IllegalStateException( "Cannot add children to a redirected node" ); | ||||||
|  |  | ||||||
|  |         if( argument instanceof HelpingArgumentBuilder ) | ||||||
|  |         { | ||||||
|  |             children.add( (HelpingArgumentBuilder) argument ); | ||||||
|  |         } | ||||||
|  |         else if( argument instanceof LiteralArgumentBuilder ) | ||||||
|  |         { | ||||||
|  |             super.then( argument ); | ||||||
|  |         } | ||||||
|  |         else | ||||||
|  |         { | ||||||
|  |             throw new IllegalStateException( "HelpingArgumentBuilder can only accept literal children" ); | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         return this; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     @Override | ||||||
|  |     public LiteralArgumentBuilder<CommandSource> then( CommandNode<CommandSource> argument ) | ||||||
|  |     { | ||||||
|  |         if( !(argument instanceof LiteralCommandNode) ) | ||||||
|  |         { | ||||||
|  |             throw new IllegalStateException( "HelpingArgumentBuilder can only accept literal children" ); | ||||||
|  |         } | ||||||
|  |         return super.then( argument ); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     @Override | ||||||
|  |     public LiteralCommandNode<CommandSource> build() | ||||||
|  |     { | ||||||
|  |         return buildImpl( getLiteral().replace( '-', '_' ), getLiteral() ); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     private LiteralCommandNode<CommandSource> build( @Nonnull String id, @Nonnull String command ) | ||||||
|  |     { | ||||||
|  |         return buildImpl( id + "." + getLiteral().replace( '-', '_' ), command + " " + getLiteral() ); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     private LiteralCommandNode<CommandSource> buildImpl( String id, String command ) | ||||||
|  |     { | ||||||
|  |         HelpCommand helpCommand = new HelpCommand( id, command ); | ||||||
|  |         LiteralCommandNode<CommandSource> node = new LiteralCommandNode<>( getLiteral(), helpCommand, getRequirement(), getRedirect(), getRedirectModifier(), isFork() ); | ||||||
|  |         helpCommand.node = node; | ||||||
|  |  | ||||||
|  |         // Set up a /... help command | ||||||
|  |         LiteralArgumentBuilder<CommandSource> helpNode = LiteralArgumentBuilder.<CommandSource>literal( "help" ) | ||||||
|  |             .requires( x -> getArguments().stream().anyMatch( y -> y.getRequirement().test( x ) ) ) | ||||||
|  |             .executes( helpCommand ); | ||||||
|  |  | ||||||
|  |         // Add all normal command children to this and the help node | ||||||
|  |         for( CommandNode<CommandSource> child : getArguments() ) | ||||||
|  |         { | ||||||
|  |             node.addChild( child ); | ||||||
|  |  | ||||||
|  |             helpNode.then( LiteralArgumentBuilder.<CommandSource>literal( child.getName() ) | ||||||
|  |                 .requires( child.getRequirement() ) | ||||||
|  |                 .executes( helpForChild( child, id, command ) ) | ||||||
|  |                 .build() | ||||||
|  |             ); | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         // And add alternative versions of which forward instead | ||||||
|  |         for( HelpingArgumentBuilder childBuilder : children ) | ||||||
|  |         { | ||||||
|  |             LiteralCommandNode<CommandSource> child = childBuilder.build( id, command ); | ||||||
|  |             node.addChild( child ); | ||||||
|  |             helpNode.then( LiteralArgumentBuilder.<CommandSource>literal( child.getName() ) | ||||||
|  |                 .requires( child.getRequirement() ) | ||||||
|  |                 .executes( helpForChild( child, id, command ) ) | ||||||
|  |                 .redirect( child.getChild( "help" ) ) | ||||||
|  |                 .build() | ||||||
|  |             ); | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         node.addChild( helpNode.build() ); | ||||||
|  |  | ||||||
|  |         return node; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     private static final TextFormatting HEADER = TextFormatting.LIGHT_PURPLE; | ||||||
|  |     private static final TextFormatting SYNOPSIS = TextFormatting.AQUA; | ||||||
|  |     private static final TextFormatting NAME = TextFormatting.GREEN; | ||||||
|  |  | ||||||
|  |     private static final class HelpCommand implements Command<CommandSource> | ||||||
|  |     { | ||||||
|  |         private final String id; | ||||||
|  |         private final String command; | ||||||
|  |         LiteralCommandNode<CommandSource> node; | ||||||
|  |  | ||||||
|  |         private HelpCommand( String id, String command ) | ||||||
|  |         { | ||||||
|  |             this.id = id; | ||||||
|  |             this.command = command; | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         @Override | ||||||
|  |         public int run( CommandContext<CommandSource> context ) | ||||||
|  |         { | ||||||
|  |             context.getSource().sendFeedback( getHelp( context, node, id, command ), false ); | ||||||
|  |             return 0; | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     private static Command<CommandSource> helpForChild( CommandNode<CommandSource> node, String id, String command ) | ||||||
|  |     { | ||||||
|  |         return context -> { | ||||||
|  |             context.getSource().sendFeedback( getHelp( context, node, id + "." + node.getName().replace( '-', '_' ), command + " " + node.getName() ), false ); | ||||||
|  |             return 0; | ||||||
|  |         }; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     private static ITextComponent getHelp( CommandContext<CommandSource> context, CommandNode<CommandSource> node, String id, String command ) | ||||||
|  |     { | ||||||
|  |         // An ugly hack to extract usage information from the dispatcher. We generate a temporary node, generate | ||||||
|  |         // the shorthand usage, and emit that. | ||||||
|  |         CommandDispatcher<CommandSource> dispatcher = context.getSource().getServer().getCommandManager().getDispatcher(); | ||||||
|  |         CommandNode<CommandSource> temp = new LiteralCommandNode<>( "_", null, x -> true, null, null, false ); | ||||||
|  |         temp.addChild( node ); | ||||||
|  |         String usage = dispatcher.getSmartUsage( temp, context.getSource() ).get( node ).substring( node.getName().length() ); | ||||||
|  |  | ||||||
|  |         ITextComponent output = new TextComponentString( "" ) | ||||||
|  |             .appendSibling( coloured( "/" + command + usage, HEADER ) ) | ||||||
|  |             .appendText( " " ) | ||||||
|  |             .appendSibling( coloured( translate( "commands." + id + ".synopsis" ), SYNOPSIS ) ) | ||||||
|  |             .appendText( "\n" ) | ||||||
|  |             .appendSibling( translate( "commands." + id + ".desc" ) ); | ||||||
|  |  | ||||||
|  |         for( CommandNode<CommandSource> child : node.getChildren() ) | ||||||
|  |         { | ||||||
|  |             if( !child.getRequirement().test( context.getSource() ) || !(child instanceof LiteralCommandNode) ) | ||||||
|  |             { | ||||||
|  |                 continue; | ||||||
|  |             } | ||||||
|  |  | ||||||
|  |             output.appendText( "\n" ); | ||||||
|  |  | ||||||
|  |             ITextComponent component = coloured( child.getName(), NAME ); | ||||||
|  |             component.getStyle().setClickEvent( new ClickEvent( | ||||||
|  |                 ClickEvent.Action.SUGGEST_COMMAND, | ||||||
|  |                 "/" + command + " " + child.getName() | ||||||
|  |             ) ); | ||||||
|  |             output.appendSibling( component ); | ||||||
|  |  | ||||||
|  |             output.appendText( " - " ).appendSibling( translate( "commands." + id + "." + child.getName() + ".synopsis" ) ); | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         return output; | ||||||
|  |     } | ||||||
|  | } | ||||||
| @@ -1,107 +0,0 @@ | |||||||
| /* |  | ||||||
|  * This file is part of ComputerCraft - http://www.computercraft.info |  | ||||||
|  * Copyright Daniel Ratcliffe, 2011-2019. Do not distribute without permission. |  | ||||||
|  * Send enquiries to dratcliffe@gmail.com |  | ||||||
|  */ |  | ||||||
|  |  | ||||||
| package dan200.computercraft.shared.command.framework; |  | ||||||
|  |  | ||||||
| import com.google.common.collect.Lists; |  | ||||||
| import dan200.computercraft.shared.util.StringUtil; |  | ||||||
| import net.minecraft.command.ICommandSender; |  | ||||||
| import net.minecraft.entity.player.EntityPlayerMP; |  | ||||||
| import net.minecraft.server.MinecraftServer; |  | ||||||
| import net.minecraftforge.common.util.FakePlayer; |  | ||||||
|  |  | ||||||
| import java.util.Collections; |  | ||||||
| import java.util.List; |  | ||||||
|  |  | ||||||
| /** |  | ||||||
|  * Represents the way a command was invoked, including the command sender, the current server and |  | ||||||
|  * the "path" to this command. |  | ||||||
|  */ |  | ||||||
| public final class CommandContext |  | ||||||
| { |  | ||||||
|     private final MinecraftServer server; |  | ||||||
|     private final ICommandSender sender; |  | ||||||
|     private final List<ISubCommand> path; |  | ||||||
|  |  | ||||||
|     public CommandContext( MinecraftServer server, ICommandSender sender, ISubCommand initial ) |  | ||||||
|     { |  | ||||||
|         this.server = server; |  | ||||||
|         this.sender = sender; |  | ||||||
|         path = Collections.singletonList( initial ); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     private CommandContext( MinecraftServer server, ICommandSender sender, List<ISubCommand> path ) |  | ||||||
|     { |  | ||||||
|         this.server = server; |  | ||||||
|         this.sender = sender; |  | ||||||
|         this.path = path; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     public CommandContext enter( ISubCommand child ) |  | ||||||
|     { |  | ||||||
|         List<ISubCommand> newPath = Lists.newArrayListWithExpectedSize( path.size() + 1 ); |  | ||||||
|         newPath.addAll( path ); |  | ||||||
|         newPath.add( child ); |  | ||||||
|         return new CommandContext( server, sender, newPath ); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     public CommandContext parent() |  | ||||||
|     { |  | ||||||
|         if( path.size() == 1 ) throw new IllegalStateException( "No parent command" ); |  | ||||||
|         return new CommandContext( server, sender, path.subList( 0, path.size() - 1 ) ); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     public String getFullPath() |  | ||||||
|     { |  | ||||||
|         StringBuilder out = new StringBuilder(); |  | ||||||
|         boolean first = true; |  | ||||||
|         for( ISubCommand command : path ) |  | ||||||
|         { |  | ||||||
|             if( first ) |  | ||||||
|             { |  | ||||||
|                 first = false; |  | ||||||
|             } |  | ||||||
|             else |  | ||||||
|             { |  | ||||||
|                 out.append( ' ' ); |  | ||||||
|             } |  | ||||||
|  |  | ||||||
|             out.append( command.getName() ); |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         return out.toString(); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     public String getFullUsage() |  | ||||||
|     { |  | ||||||
|         return "/" + getFullPath() + " " + StringUtil.translate( path.get( path.size() - 1 ).getUsage( this ) ); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     public List<ISubCommand> getPath() |  | ||||||
|     { |  | ||||||
|         return Collections.unmodifiableList( path ); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     public String getRootCommand() |  | ||||||
|     { |  | ||||||
|         return path.get( 0 ).getName(); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     public MinecraftServer getServer() |  | ||||||
|     { |  | ||||||
|         return server; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     public ICommandSender getSender() |  | ||||||
|     { |  | ||||||
|         return sender; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     public boolean fromPlayer() |  | ||||||
|     { |  | ||||||
|         return sender instanceof EntityPlayerMP && !(sender instanceof FakePlayer); |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| @@ -1,97 +0,0 @@ | |||||||
| /* |  | ||||||
|  * This file is part of ComputerCraft - http://www.computercraft.info |  | ||||||
|  * Copyright Daniel Ratcliffe, 2011-2019. Do not distribute without permission. |  | ||||||
|  * Send enquiries to dratcliffe@gmail.com |  | ||||||
|  */ |  | ||||||
|  |  | ||||||
| package dan200.computercraft.shared.command.framework; |  | ||||||
|  |  | ||||||
| import dan200.computercraft.ComputerCraft; |  | ||||||
| import net.minecraft.command.CommandException; |  | ||||||
| import net.minecraft.command.ICommand; |  | ||||||
| import net.minecraft.command.ICommandSender; |  | ||||||
| import net.minecraft.server.MinecraftServer; |  | ||||||
| import net.minecraft.util.math.BlockPos; |  | ||||||
|  |  | ||||||
| import javax.annotation.Nonnull; |  | ||||||
| import javax.annotation.Nullable; |  | ||||||
| import java.util.Arrays; |  | ||||||
| import java.util.Collections; |  | ||||||
| import java.util.List; |  | ||||||
|  |  | ||||||
| /** |  | ||||||
|  * {@link ICommand} which delegates to a {@link ISubCommand}. |  | ||||||
|  */ |  | ||||||
| public class CommandDelegate implements ICommand |  | ||||||
| { |  | ||||||
|     private final ISubCommand command; |  | ||||||
|  |  | ||||||
|     public CommandDelegate( ISubCommand command ) |  | ||||||
|     { |  | ||||||
|         this.command = command; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     @Nonnull |  | ||||||
|     @Override |  | ||||||
|     public String getName() |  | ||||||
|     { |  | ||||||
|         return command.getName(); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     @Nonnull |  | ||||||
|     @Override |  | ||||||
|     public String getUsage( @Nonnull ICommandSender sender ) |  | ||||||
|     { |  | ||||||
|         return new CommandContext( sender.getServer(), sender, command ).getFullUsage(); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     @Nonnull |  | ||||||
|     @Override |  | ||||||
|     public List<String> getAliases() |  | ||||||
|     { |  | ||||||
|         return Collections.emptyList(); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     @Override |  | ||||||
|     public void execute( @Nonnull MinecraftServer server, @Nonnull ICommandSender sender, @Nonnull String[] args ) throws CommandException |  | ||||||
|     { |  | ||||||
|         try |  | ||||||
|         { |  | ||||||
|             command.execute( new CommandContext( server, sender, command ), Arrays.asList( args ) ); |  | ||||||
|         } |  | ||||||
|         catch( CommandException e ) |  | ||||||
|         { |  | ||||||
|             throw e; |  | ||||||
|         } |  | ||||||
|         catch( Throwable e ) |  | ||||||
|         { |  | ||||||
|             ComputerCraft.log.error( "Unhandled exception in command", e ); |  | ||||||
|             throw new CommandException( "commands.computercraft.generic.exception", e.toString() ); |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     @Nonnull |  | ||||||
|     @Override |  | ||||||
|     public List<String> getTabCompletions( @Nonnull MinecraftServer server, @Nonnull ICommandSender sender, @Nonnull String[] args, @Nullable BlockPos pos ) |  | ||||||
|     { |  | ||||||
|         return command.getCompletion( new CommandContext( server, sender, command ), Arrays.asList( args ) ); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     @Override |  | ||||||
|     public boolean checkPermission( @Nonnull MinecraftServer server, @Nonnull ICommandSender sender ) |  | ||||||
|     { |  | ||||||
|         return command.checkPermission( new CommandContext( server, sender, command ) ); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     @Override |  | ||||||
|     public boolean isUsernameIndex( @Nonnull String[] args, int index ) |  | ||||||
|     { |  | ||||||
|         return false; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     @Override |  | ||||||
|     public int compareTo( @Nonnull ICommand o ) |  | ||||||
|     { |  | ||||||
|         return getName().compareTo( o.getName() ); |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| @@ -1,161 +0,0 @@ | |||||||
| /* |  | ||||||
|  * This file is part of ComputerCraft - http://www.computercraft.info |  | ||||||
|  * Copyright Daniel Ratcliffe, 2011-2019. Do not distribute without permission. |  | ||||||
|  * Send enquiries to dratcliffe@gmail.com |  | ||||||
|  */ |  | ||||||
|  |  | ||||||
| package dan200.computercraft.shared.command.framework; |  | ||||||
|  |  | ||||||
| import com.google.common.collect.Lists; |  | ||||||
| import com.google.common.collect.Maps; |  | ||||||
| import dan200.computercraft.shared.command.text.ChatHelpers; |  | ||||||
| import net.minecraft.command.CommandBase; |  | ||||||
| import net.minecraft.command.CommandException; |  | ||||||
|  |  | ||||||
| import javax.annotation.Nonnull; |  | ||||||
| import java.util.Collections; |  | ||||||
| import java.util.List; |  | ||||||
| import java.util.Map; |  | ||||||
|  |  | ||||||
| /** |  | ||||||
|  * A command which delegates to a series of sub commands |  | ||||||
|  */ |  | ||||||
| public class CommandRoot implements ISubCommand |  | ||||||
| { |  | ||||||
|     private final String name; |  | ||||||
|     private ISubCommand parent; |  | ||||||
|     private final Map<String, ISubCommand> subCommands = Maps.newHashMap(); |  | ||||||
|  |  | ||||||
|     public CommandRoot( String name ) |  | ||||||
|     { |  | ||||||
|         this.name = name; |  | ||||||
|         register( new SubCommandHelp( this ) ); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     public CommandRoot register( ISubCommand command ) |  | ||||||
|     { |  | ||||||
|         subCommands.put( command.getName(), command ); |  | ||||||
|         if( command instanceof SubCommandBase ) |  | ||||||
|         { |  | ||||||
|             ((SubCommandBase) command).setParent( this ); |  | ||||||
|         } |  | ||||||
|         else if( command instanceof CommandRoot ) |  | ||||||
|         { |  | ||||||
|             ((CommandRoot) command).setParent( this ); |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         return this; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     @Nonnull |  | ||||||
|     @Override |  | ||||||
|     public String getName() |  | ||||||
|     { |  | ||||||
|         return name; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     @Nonnull |  | ||||||
|     @Override |  | ||||||
|     public String getFullName() |  | ||||||
|     { |  | ||||||
|         return parent == null ? name : parent.getFullName() + "." + name; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     @Nonnull |  | ||||||
|     @Override |  | ||||||
|     public String getUsage( CommandContext context ) |  | ||||||
|     { |  | ||||||
|         StringBuilder out = new StringBuilder( "<" ); |  | ||||||
|         boolean first = true; |  | ||||||
|         for( ISubCommand command : subCommands.values() ) |  | ||||||
|         { |  | ||||||
|             if( command.checkPermission( context ) ) |  | ||||||
|             { |  | ||||||
|                 if( first ) |  | ||||||
|                 { |  | ||||||
|                     first = false; |  | ||||||
|                 } |  | ||||||
|                 else |  | ||||||
|                 { |  | ||||||
|                     out.append( "|" ); |  | ||||||
|                 } |  | ||||||
|  |  | ||||||
|                 out.append( command.getName() ); |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         return out.append( ">" ).toString(); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     @Override |  | ||||||
|     public boolean checkPermission( @Nonnull CommandContext context ) |  | ||||||
|     { |  | ||||||
|         for( ISubCommand command : subCommands.values() ) |  | ||||||
|         { |  | ||||||
|             if( !(command instanceof SubCommandHelp) && command.checkPermission( context ) ) return true; |  | ||||||
|         } |  | ||||||
|         return false; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     public Map<String, ISubCommand> getSubCommands() |  | ||||||
|     { |  | ||||||
|         return Collections.unmodifiableMap( subCommands ); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     @Override |  | ||||||
|     public void execute( @Nonnull CommandContext context, @Nonnull List<String> arguments ) throws CommandException |  | ||||||
|     { |  | ||||||
|         if( arguments.isEmpty() ) |  | ||||||
|         { |  | ||||||
|             context.getSender().sendMessage( ChatHelpers.getHelp( context, this, context.getFullPath() ) ); |  | ||||||
|         } |  | ||||||
|         else |  | ||||||
|         { |  | ||||||
|             ISubCommand command = subCommands.get( arguments.get( 0 ) ); |  | ||||||
|             if( command == null || !command.checkPermission( context ) ) |  | ||||||
|             { |  | ||||||
|                 throw new CommandException( context.getFullUsage() ); |  | ||||||
|             } |  | ||||||
|  |  | ||||||
|             command.execute( context.enter( command ), arguments.subList( 1, arguments.size() ) ); |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     @Nonnull |  | ||||||
|     @Override |  | ||||||
|     public List<String> getCompletion( @Nonnull CommandContext context, @Nonnull List<String> arguments ) |  | ||||||
|     { |  | ||||||
|         if( arguments.isEmpty() ) |  | ||||||
|         { |  | ||||||
|             return Lists.newArrayList( subCommands.keySet() ); |  | ||||||
|         } |  | ||||||
|         else if( arguments.size() == 1 ) |  | ||||||
|         { |  | ||||||
|             List<String> list = Lists.newArrayList(); |  | ||||||
|             String match = arguments.get( 0 ); |  | ||||||
|  |  | ||||||
|             for( ISubCommand command : subCommands.values() ) |  | ||||||
|             { |  | ||||||
|                 if( CommandBase.doesStringStartWith( match, command.getName() ) && command.checkPermission( context ) ) |  | ||||||
|                 { |  | ||||||
|                     list.add( command.getName() ); |  | ||||||
|                 } |  | ||||||
|             } |  | ||||||
|  |  | ||||||
|             return list; |  | ||||||
|         } |  | ||||||
|         else |  | ||||||
|         { |  | ||||||
|             ISubCommand command = subCommands.get( arguments.get( 0 ) ); |  | ||||||
|             if( command == null || !command.checkPermission( context ) ) return Collections.emptyList(); |  | ||||||
|  |  | ||||||
|             return command.getCompletion( context, arguments.subList( 1, arguments.size() ) ); |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     void setParent( @Nonnull ISubCommand parent ) |  | ||||||
|     { |  | ||||||
|         if( this.parent != null ) throw new IllegalStateException( "Cannot have multiple parents" ); |  | ||||||
|         this.parent = parent; |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| @@ -1,86 +0,0 @@ | |||||||
| /* |  | ||||||
|  * This file is part of ComputerCraft - http://www.computercraft.info |  | ||||||
|  * Copyright Daniel Ratcliffe, 2011-2019. Do not distribute without permission. |  | ||||||
|  * Send enquiries to dratcliffe@gmail.com |  | ||||||
|  */ |  | ||||||
|  |  | ||||||
| package dan200.computercraft.shared.command.framework; |  | ||||||
|  |  | ||||||
| import net.minecraft.command.CommandException; |  | ||||||
| import net.minecraft.command.ICommand; |  | ||||||
| import net.minecraft.command.ICommandSender; |  | ||||||
| import net.minecraft.server.MinecraftServer; |  | ||||||
| import net.minecraft.util.math.BlockPos; |  | ||||||
|  |  | ||||||
| import javax.annotation.Nonnull; |  | ||||||
| import java.util.Collections; |  | ||||||
| import java.util.List; |  | ||||||
|  |  | ||||||
| /** |  | ||||||
|  * A slightly different implementation of {@link ICommand} which is delegated to. |  | ||||||
|  */ |  | ||||||
| public interface ISubCommand |  | ||||||
| { |  | ||||||
|     /** |  | ||||||
|      * Get the name of this command |  | ||||||
|      * |  | ||||||
|      * @return The name of this command |  | ||||||
|      * @see ICommand#getName() |  | ||||||
|      */ |  | ||||||
|     @Nonnull |  | ||||||
|     String getName(); |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Get the full name of this command. This is equal to the command parent's full name, plus this command's name. |  | ||||||
|      * |  | ||||||
|      * @return The full name of this command |  | ||||||
|      * @see ISubCommand#getName() |  | ||||||
|      */ |  | ||||||
|     @Nonnull |  | ||||||
|     String getFullName(); |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Get the usage of this command |  | ||||||
|      * |  | ||||||
|      * @param context The context this command is executed in |  | ||||||
|      * @return The usage of this command |  | ||||||
|      * @see ICommand#getUsage(ICommandSender) |  | ||||||
|      */ |  | ||||||
|     @Nonnull |  | ||||||
|     default String getUsage( CommandContext context ) |  | ||||||
|     { |  | ||||||
|         return "commands." + getFullName() + ".usage"; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Determine whether a given command sender has permission to execute this command. |  | ||||||
|      * |  | ||||||
|      * @param context The current command context. |  | ||||||
|      * @return Whether this command can be executed. |  | ||||||
|      * @see ICommand#checkPermission(MinecraftServer, ICommandSender) |  | ||||||
|      */ |  | ||||||
|     boolean checkPermission( @Nonnull CommandContext context ); |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Execute this command |  | ||||||
|      * |  | ||||||
|      * @param context   The current command context. |  | ||||||
|      * @param arguments The arguments passed  @throws CommandException When an error occurs |  | ||||||
|      * @see ICommand#execute(MinecraftServer, ICommandSender, String[]) |  | ||||||
|      */ |  | ||||||
|     void execute( @Nonnull CommandContext context, @Nonnull List<String> arguments ) throws CommandException; |  | ||||||
|  |  | ||||||
|     /** |  | ||||||
|      * Get a list of possible completions |  | ||||||
|      * |  | ||||||
|      * @param context   The current command context. |  | ||||||
|      * @param arguments The arguments passed. You should complete the last one. |  | ||||||
|      * @return List of possible completions |  | ||||||
|      * @see ICommand#getTabCompletions(MinecraftServer, ICommandSender, String[], BlockPos) |  | ||||||
|      */ |  | ||||||
|     @Nonnull |  | ||||||
|     default List<String> getCompletion( @Nonnull CommandContext context, @Nonnull List<String> arguments ) |  | ||||||
|     { |  | ||||||
|         return Collections.emptyList(); |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| @@ -1,52 +0,0 @@ | |||||||
| /* |  | ||||||
|  * This file is part of ComputerCraft - http://www.computercraft.info |  | ||||||
|  * Copyright Daniel Ratcliffe, 2011-2019. Do not distribute without permission. |  | ||||||
|  * Send enquiries to dratcliffe@gmail.com |  | ||||||
|  */ |  | ||||||
|  |  | ||||||
| package dan200.computercraft.shared.command.framework; |  | ||||||
|  |  | ||||||
| import dan200.computercraft.shared.command.UserLevel; |  | ||||||
|  |  | ||||||
| import javax.annotation.Nonnull; |  | ||||||
|  |  | ||||||
| public abstract class SubCommandBase implements ISubCommand |  | ||||||
| { |  | ||||||
|     private final String name; |  | ||||||
|     private final String id; |  | ||||||
|     private final UserLevel level; |  | ||||||
|     private ISubCommand parent; |  | ||||||
|  |  | ||||||
|     protected SubCommandBase( String name, UserLevel level ) |  | ||||||
|     { |  | ||||||
|         this.name = name; |  | ||||||
|         id = name.replace( '-', '_' ); |  | ||||||
|         this.level = level; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     @Nonnull |  | ||||||
|     @Override |  | ||||||
|     public String getName() |  | ||||||
|     { |  | ||||||
|         return name; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     @Nonnull |  | ||||||
|     @Override |  | ||||||
|     public String getFullName() |  | ||||||
|     { |  | ||||||
|         return parent == null ? id : parent.getFullName() + "." + id; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     @Override |  | ||||||
|     public boolean checkPermission( @Nonnull CommandContext context ) |  | ||||||
|     { |  | ||||||
|         return level.canExecute( context ); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     void setParent( ISubCommand parent ) |  | ||||||
|     { |  | ||||||
|         if( this.parent != null ) throw new IllegalStateException( "Cannot have multiple parents" ); |  | ||||||
|         this.parent = parent; |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| @@ -1,120 +0,0 @@ | |||||||
| /* |  | ||||||
|  * This file is part of ComputerCraft - http://www.computercraft.info |  | ||||||
|  * Copyright Daniel Ratcliffe, 2011-2019. Do not distribute without permission. |  | ||||||
|  * Send enquiries to dratcliffe@gmail.com |  | ||||||
|  */ |  | ||||||
|  |  | ||||||
| package dan200.computercraft.shared.command.framework; |  | ||||||
|  |  | ||||||
| import com.google.common.collect.Lists; |  | ||||||
| import dan200.computercraft.shared.command.text.ChatHelpers; |  | ||||||
| import joptsimple.internal.Strings; |  | ||||||
| import net.minecraft.command.CommandBase; |  | ||||||
| import net.minecraft.command.CommandException; |  | ||||||
|  |  | ||||||
| import javax.annotation.Nonnull; |  | ||||||
| import java.util.Collections; |  | ||||||
| import java.util.List; |  | ||||||
|  |  | ||||||
| class SubCommandHelp implements ISubCommand |  | ||||||
| { |  | ||||||
|     private final CommandRoot branchCommand; |  | ||||||
|  |  | ||||||
|     SubCommandHelp( CommandRoot branchCommand ) |  | ||||||
|     { |  | ||||||
|         this.branchCommand = branchCommand; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     @Nonnull |  | ||||||
|     @Override |  | ||||||
|     public String getName() |  | ||||||
|     { |  | ||||||
|         return "help"; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     @Nonnull |  | ||||||
|     @Override |  | ||||||
|     public String getFullName() |  | ||||||
|     { |  | ||||||
|         return "computercraft.help"; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     @Override |  | ||||||
|     public boolean checkPermission( @Nonnull CommandContext context ) |  | ||||||
|     { |  | ||||||
|         return true; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     @Override |  | ||||||
|     public void execute( @Nonnull CommandContext context, @Nonnull List<String> arguments ) throws CommandException |  | ||||||
|     { |  | ||||||
|         ISubCommand command = branchCommand; |  | ||||||
|  |  | ||||||
|         for( int i = 0; i < arguments.size(); i++ ) |  | ||||||
|         { |  | ||||||
|             String commandName = arguments.get( i ); |  | ||||||
|             if( command instanceof CommandRoot ) |  | ||||||
|             { |  | ||||||
|                 command = ((CommandRoot) command).getSubCommands().get( commandName ); |  | ||||||
|             } |  | ||||||
|             else |  | ||||||
|             { |  | ||||||
|                 throw new CommandException( "commands.computercraft.help.no_children", Strings.join( arguments.subList( 0, i ), " " ) ); |  | ||||||
|             } |  | ||||||
|  |  | ||||||
|             if( command == null ) |  | ||||||
|             { |  | ||||||
|                 throw new CommandException( "commands.computercraft.help.no_command", Strings.join( arguments.subList( 0, i + 1 ), "  " ) ); |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         StringBuilder prefix = new StringBuilder( context.parent().getFullPath() ); |  | ||||||
|         for( String argument : arguments ) |  | ||||||
|         { |  | ||||||
|             prefix.append( ' ' ).append( argument ); |  | ||||||
|         } |  | ||||||
|         context.getSender().sendMessage( ChatHelpers.getHelp( context, command, prefix.toString() ) ); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     @Nonnull |  | ||||||
|     @Override |  | ||||||
|     public List<String> getCompletion( @Nonnull CommandContext context, @Nonnull List<String> arguments ) |  | ||||||
|     { |  | ||||||
|         CommandRoot command = branchCommand; |  | ||||||
|  |  | ||||||
|         for( int i = 0; i < arguments.size() - 1; i++ ) |  | ||||||
|         { |  | ||||||
|             String commandName = arguments.get( i ); |  | ||||||
|             ISubCommand subCommand = command.getSubCommands().get( commandName ); |  | ||||||
|  |  | ||||||
|             if( subCommand instanceof CommandRoot ) |  | ||||||
|             { |  | ||||||
|                 command = (CommandRoot) subCommand; |  | ||||||
|             } |  | ||||||
|             else |  | ||||||
|             { |  | ||||||
|                 return Collections.emptyList(); |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         if( arguments.isEmpty() ) |  | ||||||
|         { |  | ||||||
|             return Lists.newArrayList( command.getSubCommands().keySet() ); |  | ||||||
|         } |  | ||||||
|         else |  | ||||||
|         { |  | ||||||
|             List<String> list = Lists.newArrayList(); |  | ||||||
|             String match = arguments.get( arguments.size() - 1 ); |  | ||||||
|  |  | ||||||
|             for( String entry : command.getSubCommands().keySet() ) |  | ||||||
|             { |  | ||||||
|                 if( CommandBase.doesStringStartWith( match, entry ) ) |  | ||||||
|                 { |  | ||||||
|                     list.add( entry ); |  | ||||||
|                 } |  | ||||||
|             } |  | ||||||
|  |  | ||||||
|             return list; |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| @@ -6,9 +6,6 @@ | |||||||
|  |  | ||||||
| package dan200.computercraft.shared.command.text; | package dan200.computercraft.shared.command.text; | ||||||
|  |  | ||||||
| import dan200.computercraft.shared.command.framework.CommandContext; |  | ||||||
| import dan200.computercraft.shared.command.framework.CommandRoot; |  | ||||||
| import dan200.computercraft.shared.command.framework.ISubCommand; |  | ||||||
| import net.minecraft.util.math.BlockPos; | import net.minecraft.util.math.BlockPos; | ||||||
| import net.minecraft.util.text.*; | import net.minecraft.util.text.*; | ||||||
| import net.minecraft.util.text.event.ClickEvent; | import net.minecraft.util.text.event.ClickEvent; | ||||||
| @@ -20,8 +17,6 @@ import net.minecraft.util.text.event.HoverEvent; | |||||||
| public final class ChatHelpers | public final class ChatHelpers | ||||||
| { | { | ||||||
|     private static final TextFormatting HEADER = TextFormatting.LIGHT_PURPLE; |     private static final TextFormatting HEADER = TextFormatting.LIGHT_PURPLE; | ||||||
|     private static final TextFormatting SYNOPSIS = TextFormatting.AQUA; |  | ||||||
|     private static final TextFormatting NAME = TextFormatting.GREEN; |  | ||||||
|  |  | ||||||
|     private ChatHelpers() {} |     private ChatHelpers() {} | ||||||
|  |  | ||||||
| @@ -63,41 +58,6 @@ public final class ChatHelpers | |||||||
|         return component; |         return component; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     public static ITextComponent getHelp( CommandContext context, ISubCommand command, String prefix ) |  | ||||||
|     { |  | ||||||
|  |  | ||||||
|         ITextComponent output = new TextComponentString( "" ) |  | ||||||
|             .appendSibling( |  | ||||||
|                 coloured( "/" + prefix, HEADER ) |  | ||||||
|                     .appendSibling( translate( command.getUsage( context ) ) ) |  | ||||||
|             ) |  | ||||||
|             .appendText( " " ) |  | ||||||
|             .appendSibling( coloured( translate( "commands." + command.getFullName() + ".synopsis" ), SYNOPSIS ) ) |  | ||||||
|             .appendText( "\n" ) |  | ||||||
|             .appendSibling( translate( "commands." + command.getFullName() + ".desc" ) ); |  | ||||||
|  |  | ||||||
|         if( command instanceof CommandRoot ) |  | ||||||
|         { |  | ||||||
|             for( ISubCommand subCommand : ((CommandRoot) command).getSubCommands().values() ) |  | ||||||
|             { |  | ||||||
|                 if( !subCommand.checkPermission( context ) ) continue; |  | ||||||
|  |  | ||||||
|                 output.appendText( "\n" ); |  | ||||||
|  |  | ||||||
|                 ITextComponent component = coloured( subCommand.getName(), NAME ); |  | ||||||
|                 component.getStyle().setClickEvent( new ClickEvent( |  | ||||||
|                     ClickEvent.Action.SUGGEST_COMMAND, |  | ||||||
|                     "/" + prefix + " " + subCommand.getName() |  | ||||||
|                 ) ); |  | ||||||
|                 output.appendSibling( component ); |  | ||||||
|  |  | ||||||
|                 output.appendText( " - " ).appendSibling( translate( "commands." + subCommand.getFullName() + ".synopsis" ) ); |  | ||||||
|             } |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         return output; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     public static ITextComponent position( BlockPos pos ) |     public static ITextComponent position( BlockPos pos ) | ||||||
|     { |     { | ||||||
|         if( pos == null ) return translate( "commands.computercraft.generic.no_position" ); |         if( pos == null ) return translate( "commands.computercraft.generic.no_position" ); | ||||||
|   | |||||||
| @@ -6,7 +6,7 @@ | |||||||
|  |  | ||||||
| package dan200.computercraft.shared.command.text; | package dan200.computercraft.shared.command.text; | ||||||
|  |  | ||||||
| import net.minecraft.command.ICommandSender; | import net.minecraft.command.CommandSource; | ||||||
| import net.minecraft.util.text.ITextComponent; | import net.minecraft.util.text.ITextComponent; | ||||||
| import net.minecraft.util.text.TextComponentString; | import net.minecraft.util.text.TextComponentString; | ||||||
| import org.apache.commons.lang3.StringUtils; | import org.apache.commons.lang3.StringUtils; | ||||||
| @@ -15,9 +15,9 @@ import javax.annotation.Nullable; | |||||||
|  |  | ||||||
| public class ServerTableFormatter implements TableFormatter | public class ServerTableFormatter implements TableFormatter | ||||||
| { | { | ||||||
|     private final ICommandSender source; |     private final CommandSource source; | ||||||
|  |  | ||||||
|     public ServerTableFormatter( ICommandSender source ) |     public ServerTableFormatter( CommandSource source ) | ||||||
|     { |     { | ||||||
|         this.source = source; |         this.source = source; | ||||||
|     } |     } | ||||||
| @@ -40,12 +40,12 @@ public class ServerTableFormatter implements TableFormatter | |||||||
|     @Override |     @Override | ||||||
|     public int getWidth( ITextComponent component ) |     public int getWidth( ITextComponent component ) | ||||||
|     { |     { | ||||||
|         return component.getUnformattedText().length(); |         return component.getString().length(); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @Override |     @Override | ||||||
|     public void writeLine( int id, ITextComponent component ) |     public void writeLine( int id, ITextComponent component ) | ||||||
|     { |     { | ||||||
|         source.sendMessage( component ); |         source.sendFeedback( component, false ); | ||||||
|     } |     } | ||||||
| } | } | ||||||
|   | |||||||
| @@ -9,7 +9,7 @@ package dan200.computercraft.shared.command.text; | |||||||
| import dan200.computercraft.shared.command.CommandUtils; | import dan200.computercraft.shared.command.CommandUtils; | ||||||
| import dan200.computercraft.shared.network.NetworkHandler; | import dan200.computercraft.shared.network.NetworkHandler; | ||||||
| import dan200.computercraft.shared.network.client.ChatTableClientMessage; | import dan200.computercraft.shared.network.client.ChatTableClientMessage; | ||||||
| import net.minecraft.command.ICommandSender; | import net.minecraft.command.CommandSource; | ||||||
| import net.minecraft.entity.player.EntityPlayerMP; | import net.minecraft.entity.player.EntityPlayerMP; | ||||||
| import net.minecraft.util.text.ITextComponent; | import net.minecraft.util.text.ITextComponent; | ||||||
|  |  | ||||||
| @@ -120,12 +120,12 @@ public class TableBuilder | |||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     public void display( ICommandSender source ) |     public void display( CommandSource source ) | ||||||
|     { |     { | ||||||
|         if( CommandUtils.isPlayer( source ) ) |         if( CommandUtils.isPlayer( source ) ) | ||||||
|         { |         { | ||||||
|             trim( 18 ); |             trim( 18 ); | ||||||
|             NetworkHandler.sendToPlayer( (EntityPlayerMP) source, new ChatTableClientMessage( this ) ); |             NetworkHandler.sendToPlayer( (EntityPlayerMP) source.getEntity(), new ChatTableClientMessage( this ) ); | ||||||
|         } |         } | ||||||
|         else |         else | ||||||
|         { |         { | ||||||
|   | |||||||
| @@ -93,7 +93,7 @@ public interface TableFormatter | |||||||
|             // it a tad prettier. |             // it a tad prettier. | ||||||
|             int rowCharWidth = getWidth( HEADER ); |             int rowCharWidth = getWidth( HEADER ); | ||||||
|             int rowWidth = totalWidth / rowCharWidth + (totalWidth % rowCharWidth == 0 ? 0 : 1); |             int rowWidth = totalWidth / rowCharWidth + (totalWidth % rowCharWidth == 0 ? 0 : 1); | ||||||
|             writeLine( rowId++, coloured( StringUtils.repeat( HEADER.getUnformattedText(), rowWidth ), TextFormatting.GRAY ) ); |             writeLine( rowId++, coloured( StringUtils.repeat( HEADER.getString(), rowWidth ), TextFormatting.GRAY ) ); | ||||||
|         } |         } | ||||||
|  |  | ||||||
|         for( ITextComponent[] row : table.getRows() ) |         for( ITextComponent[] row : table.getRows() ) | ||||||
| @@ -118,4 +118,3 @@ public interface TableFormatter | |||||||
|         return rowId - table.getId(); |         return rowId - table.getId(); | ||||||
|     } |     } | ||||||
| } | } | ||||||
|  |  | ||||||
|   | |||||||
| @@ -1,43 +0,0 @@ | |||||||
| /* |  | ||||||
|  * This file is part of ComputerCraft - http://www.computercraft.info |  | ||||||
|  * Copyright Daniel Ratcliffe, 2011-2019. Do not distribute without permission. |  | ||||||
|  * Send enquiries to dratcliffe@gmail.com |  | ||||||
|  */ |  | ||||||
|  |  | ||||||
| package dan200.computercraft.shared.common; |  | ||||||
|  |  | ||||||
| import net.minecraft.block.material.Material; |  | ||||||
| import net.minecraft.tileentity.TileEntity; |  | ||||||
| import net.minecraft.util.EnumFacing; |  | ||||||
| import net.minecraft.util.math.BlockPos; |  | ||||||
| import net.minecraft.world.IBlockAccess; |  | ||||||
| import net.minecraft.world.World; |  | ||||||
|  |  | ||||||
| public abstract class BlockDirectional extends BlockGeneric |  | ||||||
| { |  | ||||||
|     protected BlockDirectional( Material material ) |  | ||||||
|     { |  | ||||||
|         super( material ); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     public EnumFacing getDirection( IBlockAccess world, BlockPos pos ) |  | ||||||
|     { |  | ||||||
|         TileEntity tile = world.getTileEntity( pos ); |  | ||||||
|         if( tile instanceof IDirectionalTile ) |  | ||||||
|         { |  | ||||||
|             IDirectionalTile directional = (IDirectionalTile) tile; |  | ||||||
|             return directional.getDirection(); |  | ||||||
|         } |  | ||||||
|         return EnumFacing.NORTH; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     public void setDirection( World world, BlockPos pos, EnumFacing dir ) |  | ||||||
|     { |  | ||||||
|         TileEntity tile = world.getTileEntity( pos ); |  | ||||||
|         if( tile instanceof IDirectionalTile ) |  | ||||||
|         { |  | ||||||
|             IDirectionalTile directional = (IDirectionalTile) tile; |  | ||||||
|             directional.setDirection( dir ); |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| @@ -8,42 +8,46 @@ package dan200.computercraft.shared.common; | |||||||
|  |  | ||||||
| import net.minecraft.block.Block; | import net.minecraft.block.Block; | ||||||
| import net.minecraft.block.ITileEntityProvider; | import net.minecraft.block.ITileEntityProvider; | ||||||
| import net.minecraft.block.material.Material; |  | ||||||
| import net.minecraft.block.state.IBlockState; | import net.minecraft.block.state.IBlockState; | ||||||
| import net.minecraft.entity.player.EntityPlayer; | import net.minecraft.entity.player.EntityPlayer; | ||||||
| import net.minecraft.tileentity.TileEntity; | import net.minecraft.tileentity.TileEntity; | ||||||
|  | import net.minecraft.tileentity.TileEntityType; | ||||||
| import net.minecraft.util.EnumFacing; | import net.minecraft.util.EnumFacing; | ||||||
| import net.minecraft.util.EnumHand; | import net.minecraft.util.EnumHand; | ||||||
| import net.minecraft.util.math.BlockPos; | import net.minecraft.util.math.BlockPos; | ||||||
| import net.minecraft.world.IBlockAccess; | import net.minecraft.world.IBlockReader; | ||||||
|  | import net.minecraft.world.IWorldReader; | ||||||
| import net.minecraft.world.World; | import net.minecraft.world.World; | ||||||
|  |  | ||||||
| import javax.annotation.Nonnull; | import javax.annotation.Nonnull; | ||||||
|  | import javax.annotation.Nullable; | ||||||
| import java.util.Random; | import java.util.Random; | ||||||
|  |  | ||||||
| public abstract class BlockGeneric extends Block implements ITileEntityProvider | public abstract class BlockGeneric extends Block implements ITileEntityProvider | ||||||
| { | { | ||||||
|     protected BlockGeneric( Material material ) |     private final TileEntityType<? extends TileGeneric> type; | ||||||
|  |  | ||||||
|  |     public BlockGeneric( Properties settings, TileEntityType<? extends TileGeneric> type ) | ||||||
|     { |     { | ||||||
|         super( material ); |         super( settings ); | ||||||
|         hasTileEntity = true; |         this.type = type; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     protected abstract TileGeneric createTile( IBlockState state ); |  | ||||||
|  |  | ||||||
|     protected abstract TileGeneric createTile( int damage ); |  | ||||||
|  |  | ||||||
|     @Override |     @Override | ||||||
|     public final void breakBlock( @Nonnull World world, @Nonnull BlockPos pos, @Nonnull IBlockState newState ) |     @Deprecated | ||||||
|  |     public final void onReplaced( @Nonnull IBlockState block, @Nonnull World world, @Nonnull BlockPos pos, IBlockState replace, boolean bool ) | ||||||
|     { |     { | ||||||
|  |         if( block.getBlock() == replace.getBlock() ) return; | ||||||
|  |  | ||||||
|         TileEntity tile = world.getTileEntity( pos ); |         TileEntity tile = world.getTileEntity( pos ); | ||||||
|         super.breakBlock( world, pos, newState ); |         super.onReplaced( block, world, pos, replace, bool ); | ||||||
|         world.removeTileEntity( pos ); |         world.removeTileEntity( pos ); | ||||||
|         if( tile instanceof TileGeneric ) ((TileGeneric) tile).destroy(); |         if( tile instanceof TileGeneric ) ((TileGeneric) tile).destroy(); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @Override |     @Override | ||||||
|     public final boolean onBlockActivated( World world, BlockPos pos, IBlockState state, EntityPlayer player, EnumHand hand, EnumFacing side, float hitX, float hitY, float hitZ ) |     @Deprecated | ||||||
|  |     public final boolean onBlockActivated( IBlockState state, World world, BlockPos pos, EntityPlayer player, EnumHand hand, EnumFacing side, float hitX, float hitY, float hitZ ) | ||||||
|     { |     { | ||||||
|         TileEntity tile = world.getTileEntity( pos ); |         TileEntity tile = world.getTileEntity( pos ); | ||||||
|         return tile instanceof TileGeneric && ((TileGeneric) tile).onActivate( player, hand, side, hitX, hitY, hitZ ); |         return tile instanceof TileGeneric && ((TileGeneric) tile).onActivate( player, hand, side, hitX, hitY, hitZ ); | ||||||
| @@ -59,80 +63,24 @@ public abstract class BlockGeneric extends Block implements ITileEntityProvider | |||||||
|     } |     } | ||||||
|  |  | ||||||
|     @Override |     @Override | ||||||
|     public final void onNeighborChange( IBlockAccess world, BlockPos pos, BlockPos neighbour ) |     public final void onNeighborChange( IBlockState state, IWorldReader world, BlockPos pos, BlockPos neighbour ) | ||||||
|     { |     { | ||||||
|         TileEntity tile = world.getTileEntity( pos ); |         TileEntity tile = world.getTileEntity( pos ); | ||||||
|         if( tile instanceof TileGeneric ) ((TileGeneric) tile).onNeighbourTileEntityChange( neighbour ); |         if( tile instanceof TileGeneric ) ((TileGeneric) tile).onNeighbourTileEntityChange( neighbour ); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @Override |     @Override | ||||||
|     public void updateTick( World world, BlockPos pos, IBlockState state, Random rand ) |     @Deprecated | ||||||
|  |     public void tick( IBlockState state, World world, BlockPos pos, Random rand ) | ||||||
|     { |     { | ||||||
|         TileEntity te = world.getTileEntity( pos ); |         TileEntity te = world.getTileEntity( pos ); | ||||||
|         if( te instanceof TileGeneric ) ((TileGeneric) te).updateTick(); |         if( te instanceof TileGeneric ) ((TileGeneric) te).blockTick(); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     @Nullable | ||||||
|     @Override |     @Override | ||||||
|     @Deprecated |     public TileEntity createNewTileEntity( @Nonnull IBlockReader world ) | ||||||
|     public final boolean canProvidePower( IBlockState state ) |  | ||||||
|     { |     { | ||||||
|         return true; |         return type.create(); | ||||||
|     } |  | ||||||
|  |  | ||||||
|     @Override |  | ||||||
|     public final boolean canConnectRedstone( IBlockState state, IBlockAccess world, BlockPos pos, EnumFacing side ) |  | ||||||
|     { |  | ||||||
|         TileEntity tile = world.getTileEntity( pos ); |  | ||||||
|         return tile instanceof TileGeneric && ((TileGeneric) tile).getRedstoneConnectivity( side ); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     @Override |  | ||||||
|     @Deprecated |  | ||||||
|     public final int getStrongPower( IBlockState state, IBlockAccess world, BlockPos pos, EnumFacing oppositeSide ) |  | ||||||
|     { |  | ||||||
|         TileEntity tile = world.getTileEntity( pos ); |  | ||||||
|         return tile instanceof TileGeneric && tile.hasWorld() ? ((TileGeneric) tile).getRedstoneOutput( oppositeSide.getOpposite() ) : 0; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     @Override |  | ||||||
|     @Deprecated |  | ||||||
|     public final int getWeakPower( IBlockState state, IBlockAccess world, BlockPos pos, EnumFacing oppositeSide ) |  | ||||||
|     { |  | ||||||
|         return getStrongPower( state, world, pos, oppositeSide ); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     public boolean getBundledRedstoneConnectivity( World world, BlockPos pos, EnumFacing side ) |  | ||||||
|     { |  | ||||||
|         TileEntity tile = world.getTileEntity( pos ); |  | ||||||
|         return tile instanceof TileGeneric && ((TileGeneric) tile).getBundledRedstoneConnectivity( side ); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     public int getBundledRedstoneOutput( World world, BlockPos pos, EnumFacing side ) |  | ||||||
|     { |  | ||||||
|         TileEntity tile = world.getTileEntity( pos ); |  | ||||||
|         return tile instanceof TileGeneric && tile.hasWorld() ? ((TileGeneric) tile).getBundledRedstoneOutput( side ) : 0; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     @Nonnull |  | ||||||
|     @Override |  | ||||||
|     public final TileEntity createTileEntity( @Nonnull World world, @Nonnull IBlockState state ) |  | ||||||
|     { |  | ||||||
|         return createTile( state ); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     @Nonnull |  | ||||||
|     @Override |  | ||||||
|     public final TileEntity createNewTileEntity( @Nonnull World world, int damage ) |  | ||||||
|     { |  | ||||||
|         return createTile( damage ); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     @Override |  | ||||||
|     @Deprecated |  | ||||||
|     public boolean isSideSolid( IBlockState state, @Nonnull IBlockAccess world, @Nonnull BlockPos pos, EnumFacing side ) |  | ||||||
|     { |  | ||||||
|         // We need to override this as the default implementation uses isNormalCube, which returns false if |  | ||||||
|         // it can provide power. |  | ||||||
|         return isFullCube( state ); |  | ||||||
|     } |     } | ||||||
| } | } | ||||||
|   | |||||||
| @@ -50,10 +50,10 @@ public class ClientTerminal implements ITerminal | |||||||
|     public void readDescription( NBTTagCompound nbt ) |     public void readDescription( NBTTagCompound nbt ) | ||||||
|     { |     { | ||||||
|         m_colour = nbt.getBoolean( "colour" ); |         m_colour = nbt.getBoolean( "colour" ); | ||||||
|         if( nbt.hasKey( "terminal" ) ) |         if( nbt.contains( "terminal" ) ) | ||||||
|         { |         { | ||||||
|             NBTTagCompound terminal = nbt.getCompoundTag( "terminal" ); |             NBTTagCompound terminal = nbt.getCompound( "terminal" ); | ||||||
|             resizeTerminal( terminal.getInteger( "term_width" ), terminal.getInteger( "term_height" ) ); |             resizeTerminal( terminal.getInt( "term_width" ), terminal.getInt( "term_height" ) ); | ||||||
|             m_terminal.readFromNBT( terminal ); |             m_terminal.readFromNBT( terminal ); | ||||||
|         } |         } | ||||||
|         else |         else | ||||||
|   | |||||||
| @@ -6,24 +6,30 @@ | |||||||
|  |  | ||||||
| package dan200.computercraft.shared.common; | package dan200.computercraft.shared.common; | ||||||
|  |  | ||||||
| import com.google.gson.JsonObject; | import dan200.computercraft.ComputerCraft; | ||||||
|  | import dan200.computercraft.shared.util.AbstractRecipe; | ||||||
| import dan200.computercraft.shared.util.Colour; | import dan200.computercraft.shared.util.Colour; | ||||||
| import dan200.computercraft.shared.util.ColourTracker; | import dan200.computercraft.shared.util.ColourTracker; | ||||||
| import dan200.computercraft.shared.util.ColourUtils; | import dan200.computercraft.shared.util.ColourUtils; | ||||||
| import net.minecraft.inventory.InventoryCrafting; | import net.minecraft.inventory.IInventory; | ||||||
|  | import net.minecraft.item.EnumDyeColor; | ||||||
| import net.minecraft.item.ItemStack; | import net.minecraft.item.ItemStack; | ||||||
| import net.minecraft.item.crafting.IRecipe; | import net.minecraft.item.crafting.IRecipeSerializer; | ||||||
|  | import net.minecraft.item.crafting.RecipeSerializers; | ||||||
|  | import net.minecraft.util.ResourceLocation; | ||||||
| import net.minecraft.world.World; | import net.minecraft.world.World; | ||||||
| import net.minecraftforge.common.crafting.IRecipeFactory; |  | ||||||
| import net.minecraftforge.common.crafting.JsonContext; |  | ||||||
| import net.minecraftforge.registries.IForgeRegistryEntry; |  | ||||||
|  |  | ||||||
| import javax.annotation.Nonnull; | import javax.annotation.Nonnull; | ||||||
|  |  | ||||||
| public class ColourableRecipe extends IForgeRegistryEntry.Impl<IRecipe> implements IRecipe | public class ColourableRecipe extends AbstractRecipe | ||||||
| { | { | ||||||
|  |     public ColourableRecipe( ResourceLocation id ) | ||||||
|  |     { | ||||||
|  |         super( id ); | ||||||
|  |     } | ||||||
|  |  | ||||||
|     @Override |     @Override | ||||||
|     public boolean matches( @Nonnull InventoryCrafting inv, @Nonnull World worldIn ) |     public boolean matches( @Nonnull IInventory inv, @Nonnull World world ) | ||||||
|     { |     { | ||||||
|         boolean hasColourable = false; |         boolean hasColourable = false; | ||||||
|         boolean hasDye = false; |         boolean hasDye = false; | ||||||
| @@ -37,7 +43,7 @@ public class ColourableRecipe extends IForgeRegistryEntry.Impl<IRecipe> implemen | |||||||
|                 if( hasColourable ) return false; |                 if( hasColourable ) return false; | ||||||
|                 hasColourable = true; |                 hasColourable = true; | ||||||
|             } |             } | ||||||
|             else if( ColourUtils.getStackColour( stack ) >= 0 ) |             else if( ColourUtils.getStackColour( stack ) != null ) | ||||||
|             { |             { | ||||||
|                 hasDye = true; |                 hasDye = true; | ||||||
|             } |             } | ||||||
| @@ -52,7 +58,7 @@ public class ColourableRecipe extends IForgeRegistryEntry.Impl<IRecipe> implemen | |||||||
|  |  | ||||||
|     @Nonnull |     @Nonnull | ||||||
|     @Override |     @Override | ||||||
|     public ItemStack getCraftingResult( @Nonnull InventoryCrafting inv ) |     public ItemStack getCraftingResult( @Nonnull IInventory inv ) | ||||||
|     { |     { | ||||||
|         ItemStack colourable = ItemStack.EMPTY; |         ItemStack colourable = ItemStack.EMPTY; | ||||||
|  |  | ||||||
| @@ -70,10 +76,10 @@ public class ColourableRecipe extends IForgeRegistryEntry.Impl<IRecipe> implemen | |||||||
|             } |             } | ||||||
|             else |             else | ||||||
|             { |             { | ||||||
|                 int index = ColourUtils.getStackColour( stack ); |                 EnumDyeColor dye = ColourUtils.getStackColour( stack ); | ||||||
|                 if( index < 0 ) continue; |                 if( dye == null ) continue; | ||||||
|  |  | ||||||
|                 Colour colour = Colour.values()[index]; |                 Colour colour = Colour.fromInt( 15 - dye.getId() ); | ||||||
|                 tracker.addColour( colour.getR(), colour.getG(), colour.getB() ); |                 tracker.addColour( colour.getR(), colour.getG(), colour.getB() ); | ||||||
|             } |             } | ||||||
|         } |         } | ||||||
| @@ -89,24 +95,13 @@ public class ColourableRecipe extends IForgeRegistryEntry.Impl<IRecipe> implemen | |||||||
|     } |     } | ||||||
|  |  | ||||||
|     @Override |     @Override | ||||||
|     public boolean isDynamic() |  | ||||||
|     { |  | ||||||
|         return true; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     @Nonnull |     @Nonnull | ||||||
|     @Override |     public IRecipeSerializer<?> getSerializer() | ||||||
|     public ItemStack getRecipeOutput() |  | ||||||
|     { |     { | ||||||
|         return ItemStack.EMPTY; |         return SERIALIZER; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     public static class Factory implements IRecipeFactory |     public static final IRecipeSerializer<?> SERIALIZER = new RecipeSerializers.SimpleSerializer<>( | ||||||
|     { |         ComputerCraft.MOD_ID + ":colour", ColourableRecipe::new | ||||||
|         @Override |     ); | ||||||
|         public IRecipe parse( JsonContext jsonContext, JsonObject jsonObject ) |  | ||||||
|         { |  | ||||||
|             return new ColourableRecipe(); |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
| } | } | ||||||
|   | |||||||
| @@ -4,7 +4,7 @@ | |||||||
|  * Send enquiries to dratcliffe@gmail.com |  * Send enquiries to dratcliffe@gmail.com | ||||||
|  */ |  */ | ||||||
| 
 | 
 | ||||||
| package dan200.computercraft.shared.media.inventory; | package dan200.computercraft.shared.common; | ||||||
| 
 | 
 | ||||||
| import dan200.computercraft.shared.util.InventoryUtil; | import dan200.computercraft.shared.util.InventoryUtil; | ||||||
| import net.minecraft.entity.player.EntityPlayer; | import net.minecraft.entity.player.EntityPlayer; | ||||||
| @@ -34,7 +34,7 @@ public class ContainerHeldItem extends Container | |||||||
|     @Override |     @Override | ||||||
|     public boolean canInteractWith( @Nonnull EntityPlayer player ) |     public boolean canInteractWith( @Nonnull EntityPlayer player ) | ||||||
|     { |     { | ||||||
|         if( !player.isEntityAlive() ) return false; |         if( !player.isAlive() ) return false; | ||||||
| 
 | 
 | ||||||
|         ItemStack stack = player.getHeldItem( m_hand ); |         ItemStack stack = player.getHeldItem( m_hand ); | ||||||
|         return stack == m_stack || !stack.isEmpty() && !m_stack.isEmpty() && stack.getItem() == m_stack.getItem(); |         return stack == m_stack || !stack.isEmpty() && !m_stack.isEmpty() && stack.getItem() == m_stack.getItem(); | ||||||
Some files were not shown because too many files have changed in this diff Show More
		Reference in New Issue
	
	Block a user
	 SquidDev
					SquidDev