mirror of
				https://github.com/SquidDev-CC/CC-Tweaked
				synced 2025-10-21 17:07:39 +00:00 
			
		
		
		
	Compare commits
	
		
			118 Commits
		
	
	
		
			v1.96.0
			...
			v1.14.4-1.
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|   | 92567b4d7e | ||
|   | 0ae70fed13 | ||
|   | 3b7300543a | ||
|   | 642351af1a | ||
|   | 121802a683 | ||
|   | 08cf55e55f | ||
|   | 3c8c0d78ef | ||
|   | c4d18aa9ca | ||
|   | 2d4a87adc9 | ||
|   | bedac71e3d | ||
|   | ee4e42e730 | ||
|   | 0de75f05dd | ||
|   | be6dd21e54 | ||
|   | 927ddb0bde | ||
|   | a8fadabaf1 | ||
|   | 44d0f78c1b | ||
|   | 38f9a015ca | ||
|   | c311cdc6f5 | ||
|   | a93e0f3284 | ||
|   | 14b3065ba4 | ||
|   | 3ea2d6a0a8 | ||
|   | c802290437 | ||
|   | f7781defe5 | ||
|   | 418420523a | ||
|   | d342a1f368 | ||
|   | 81f85361d5 | ||
|   | f1621b30ec | ||
|   | d4f6a594b6 | ||
|   | ff5ba5c131 | ||
|   | 4243f30308 | ||
|   | 813e91073d | ||
|   | 7250f22ff6 | ||
|   | db31a53bba | ||
|   | 3023f235a4 | ||
|   | 79cd8b4da5 | ||
|   | 8e4d311cd9 | ||
|   | 9bd8c86a94 | ||
|   | cbc0c1d0b6 | ||
|   | 49c37857d4 | ||
|   | b1139a4bf6 | ||
|   | 7e8559278e | ||
|   | 1e7f1c98fc | ||
|   | a802f25dd6 | ||
|   | f1d6d21d6d | ||
|   | a80302c513 | ||
|   | 1c46949da7 | ||
|   | 07a56454a0 | ||
|   | a0e72d02c8 | ||
|   | 455a59ca85 | ||
|   | 46d78af068 | ||
|   | 08d22fd3df | ||
|   | e6c691a8f8 | ||
|   | 4b0e5c445c | ||
|   | eb5cff1045 | ||
|   | 35c7792aa2 | ||
|   | 521688d630 | ||
|   | 75e2845c01 | ||
|   | 2f96283286 | ||
|   | cbe6e9b5f5 | ||
|   | 2ab79cf474 | ||
|   | 6ce34aba79 | ||
|   | 5eeb320b60 | ||
|   | 93310850d2 | ||
|   | a2880b12ca | ||
|   | cef2657048 | ||
|   | ccd85eb055 | ||
|   | 303b57779a | ||
|   | 6279816ecc | ||
|   | 4ae77261fa | ||
|   | 4b7d843b78 | ||
|   | 1c28df65c3 | ||
|   | 85b740f484 | ||
|   | f9929cb27d | ||
|   | bafab1ac07 | ||
|   | e05c262468 | ||
|   | acfb72246c | ||
|   | 9d51c4c340 | ||
|   | 18068effec | ||
|   | 7a3f7d3bba | ||
|   | 95aa48c456 | ||
|   | 6ea8ca991b | ||
|   | f1e551b960 | ||
|   | 772c54ec74 | ||
|   | 13cb789c18 | ||
|   | 42220c4268 | ||
|   | 3052506e2e | ||
|   | 0741daa7eb | ||
|   | b4aa554279 | ||
|   | 8fe2abe0ae | ||
|   | 5af789ae11 | ||
|   | 904a168d5c | ||
|   | 724441eddc | ||
|   | f68ab3edd1 | ||
|   | 68542aca3a | ||
|   | 594bc4203c | ||
|   | 57318b022d | ||
|   | 761159aa93 | ||
|   | 29dce26bf6 | ||
|   | 717ab69093 | ||
|   | 138a2cf08f | ||
|   | 81daf82647 | ||
|   | f3798bfb63 | ||
|   | bc07dfad2e | ||
|   | 8dd1c2a6cc | ||
|   | d10b657a54 | ||
|   | f90da739eb | ||
|   | d9cadf64e8 | ||
|   | 15d4a55cd8 | ||
|   | 309cbdb8be | ||
|   | 39a9ad0ce7 | ||
|   | 0f3c44c926 | ||
|   | a0e7c4a74c | ||
|   | 7d428030df | ||
|   | 00c395f689 | ||
|   | d8e1c73d26 | ||
|   | ffa4cc241b | ||
|   | 6f1b740c8f | ||
|   | 3406ba3ebf | 
							
								
								
									
										18
									
								
								.github/workflows/main-ci.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						
									
										18
									
								
								.github/workflows/main-ci.yml
									
									
									
									
										vendored
									
									
										Normal file
									
								
							| @@ -0,0 +1,18 @@ | |||||||
|  | name: Build | ||||||
|  |  | ||||||
|  | on: [push, pull_request] | ||||||
|  |  | ||||||
|  | jobs: | ||||||
|  |   build: | ||||||
|  |     runs-on: ubuntu-latest | ||||||
|  |  | ||||||
|  |     steps: | ||||||
|  |     - uses: actions/checkout@v1 | ||||||
|  |  | ||||||
|  |     - name: Set up JDK 1.8 | ||||||
|  |       uses: actions/setup-java@v1 | ||||||
|  |       with: | ||||||
|  |         java-version: 1.8 | ||||||
|  |  | ||||||
|  |     - name: Build with Gradle | ||||||
|  |       run: ./gradlew build --no-daemon | ||||||
							
								
								
									
										6
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							
							
						
						
									
										6
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							| @@ -15,3 +15,9 @@ | |||||||
| .idea | .idea | ||||||
| .gradle | .gradle | ||||||
| *.DS_Store | *.DS_Store | ||||||
|  |  | ||||||
|  | .classpath | ||||||
|  | .project | ||||||
|  | .settings/ | ||||||
|  | bin/ | ||||||
|  | *.launch | ||||||
|   | |||||||
| @@ -17,7 +17,8 @@ ignore = { | |||||||
| -- are largely unsupported. | -- are largely unsupported. | ||||||
| include_files = { | include_files = { | ||||||
|     'src/main/resources/assets/computercraft/lua/rom', |     'src/main/resources/assets/computercraft/lua/rom', | ||||||
|     'src/main/resources/assets/computercraft/lua/bios.lua' |     'src/main/resources/assets/computercraft/lua/bios.lua', | ||||||
|  |     'src/test/resources/test-rom', | ||||||
| } | } | ||||||
|  |  | ||||||
| files['src/main/resources/assets/computercraft/lua/bios.lua'] = { | files['src/main/resources/assets/computercraft/lua/bios.lua'] = { | ||||||
|   | |||||||
							
								
								
									
										14
									
								
								.travis.yml
									
									
									
									
									
								
							
							
						
						
									
										14
									
								
								.travis.yml
									
									
									
									
									
								
							| @@ -1,14 +0,0 @@ | |||||||
| language: java |  | ||||||
|  |  | ||||||
| script: ./gradlew build --no-daemon |  | ||||||
|  |  | ||||||
| before_cache: |  | ||||||
|   - rm -f  $HOME/.gradle/caches/modules-2/modules-2.lock |  | ||||||
|   - rm -fr $HOME/.gradle/caches/*/plugin-resolution/ |  | ||||||
| cache: |  | ||||||
|   directories: |  | ||||||
|     - $HOME/.gradle/caches/ |  | ||||||
|     - $HOME/.gradle/wrapper/s |  | ||||||
|  |  | ||||||
| jdk: |  | ||||||
|     - oraclejdk8 |  | ||||||
| @@ -1,5 +1,5 @@ | |||||||
| #  | #  | ||||||
| [](https://travis-ci.org/SquidDev-CC/CC-Tweaked "Current build status") [](https://minecraft.curseforge.com/projects/cc-tweaked "Download CC: Tweaked on CurseForge") | [](https://github.com/SquidDev-CC/CC-Tweaked/actions "Current build status") [](https://minecraft.curseforge.com/projects/cc-tweaked "Download CC: Tweaked on CurseForge") | ||||||
|  |  | ||||||
| CC: Tweaked is a fork of [ComputerCraft](https://github.com/dan200/ComputerCraft), adding programmable computers, | CC: Tweaked is a fork of [ComputerCraft](https://github.com/dan200/ComputerCraft), adding programmable computers, | ||||||
| turtles and more to Minecraft. | turtles and more to Minecraft. | ||||||
| @@ -41,7 +41,7 @@ Any contribution is welcome, be that using the mod, reporting bugs or contributi | |||||||
| develop CC:T, you'll need to follow these steps: | develop CC:T, you'll need to follow these steps: | ||||||
|  |  | ||||||
|  - **Clone the repository:** `git clone https://github.com/SquidDev-CC/CC-Tweaked.git && cd CC-Tweaked` |  - **Clone the repository:** `git clone https://github.com/SquidDev-CC/CC-Tweaked.git && cd CC-Tweaked` | ||||||
|  - **Setup Forge:** `./gradlew setupDecompWorkspace` |  - **Setup Forge:** `./gradlew build` | ||||||
|  - **Test your changes:** `./gradlew runClient` (or run the `GradleStart` class from your IDE). |  - **Test your changes:** `./gradlew runClient` (or run the `GradleStart` class from your IDE). | ||||||
|  |  | ||||||
| If you want to run CC:T in a normal Minecraft instance, run `./gradlew build` and copy the `.jar` from `build/libs`. | If you want to run CC:T in a normal Minecraft instance, run `./gradlew build` and copy the `.jar` from `build/libs`. | ||||||
|   | |||||||
							
								
								
									
										116
									
								
								build.gradle
									
									
									
									
									
								
							
							
						
						
									
										116
									
								
								build.gradle
									
									
									
									
									
								
							| @@ -4,19 +4,21 @@ buildscript { | |||||||
|         mavenCentral() |         mavenCentral() | ||||||
|         maven { |         maven { | ||||||
|             name = "forge" |             name = "forge" | ||||||
|             url = "http://files.minecraftforge.net/maven" |             url = "https://files.minecraftforge.net/maven" | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
|     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:3.0.117' |         classpath 'net.minecraftforge.gradle:ForgeGradle:3.0.154' | ||||||
|         classpath 'net.sf.proguard:proguard-gradle:6.1.0beta2' |         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.2.0' |     id "checkstyle" | ||||||
|  |     id "com.github.hierynomus.license" version "0.15.0" | ||||||
|  |     id "com.matthewprenger.cursegradle" version "1.3.0" | ||||||
|     id "com.github.breadmoirai.github-release" version "2.2.4" |     id "com.github.breadmoirai.github-release" version "2.2.4" | ||||||
| } | } | ||||||
|  |  | ||||||
| @@ -65,7 +67,7 @@ minecraft { | |||||||
| repositories { | repositories { | ||||||
|     maven { |     maven { | ||||||
|         name "JEI" |         name "JEI" | ||||||
|         url  "http://dvs1.progwml6.com/files/maven" |         url "https://dvs1.progwml6.com/files/maven" | ||||||
|     } |     } | ||||||
|     maven { |     maven { | ||||||
|         name "SquidDev" |         name "SquidDev" | ||||||
| @@ -77,7 +79,7 @@ repositories { | |||||||
|     } |     } | ||||||
|     maven { |     maven { | ||||||
|         name "Amadornes" |         name "Amadornes" | ||||||
|         url "http://maven.amadornes.com/" |         url "https://maven.amadornes.com/" | ||||||
|     } |     } | ||||||
| } | } | ||||||
|  |  | ||||||
| @@ -88,30 +90,25 @@ configurations { | |||||||
| } | } | ||||||
|  |  | ||||||
| dependencies { | dependencies { | ||||||
|  |     checkstyle "com.puppycrawl.tools:checkstyle:8.25" | ||||||
|  |  | ||||||
|     minecraft "net.minecraftforge:forge:${mc_version}-${forge_version}" |     minecraft "net.minecraftforge:forge:${mc_version}-${forge_version}" | ||||||
|  |  | ||||||
|     compileOnly fg.deobf("mezz.jei:jei-1.13.2:5.0.0.20:api") |     compileOnly fg.deobf("mezz.jei:jei-1.14.4:6.0.0.25:api") | ||||||
|     // deobfProvided "pl.asie:Charset-Lib:0.5.4.6" |     // deobfProvided "pl.asie:Charset-Lib:0.5.4.6" | ||||||
|     // deobfProvided "MCMultiPart2:MCMultiPart:2.5.3" |     // deobfProvided "MCMultiPart2:MCMultiPart:2.5.3" | ||||||
|  |  | ||||||
|     runtimeOnly fg.deobf("mezz.jei:jei-1.13.2:5.0.0.20") |     runtimeOnly fg.deobf("mezz.jei:jei-1.14.4:6.0.0.25") | ||||||
|  |  | ||||||
|     shade 'org.squiddev:Cobalt:0.5.0-SNAPSHOT' |     shade 'org.squiddev:Cobalt:0.5.0-SNAPSHOT' | ||||||
|  |  | ||||||
|     testImplementation 'org.junit.jupiter:junit-jupiter-api:5.1.0' |     testImplementation 'org.junit.jupiter:junit-jupiter-api:5.4.2' | ||||||
|     testRuntimeOnly 'org.junit.jupiter:junit-jupiter-engine:5.1.0' |     testRuntimeOnly 'org.junit.jupiter:junit-jupiter-engine:5.4.2' | ||||||
|  |  | ||||||
|     deployerJars "org.apache.maven.wagon:wagon-ssh:3.0.0" |     deployerJars "org.apache.maven.wagon:wagon-ssh:3.0.0" | ||||||
| } | } | ||||||
|  |  | ||||||
| sourceSets { | // Compile tasks | ||||||
|     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" | ||||||
| @@ -123,7 +120,7 @@ jar { | |||||||
|     manifest { |     manifest { | ||||||
|         attributes(["Specification-Title": "computercraft", |         attributes(["Specification-Title": "computercraft", | ||||||
|                     "Specification-Vendor": "SquidDev", |                     "Specification-Vendor": "SquidDev", | ||||||
|                     "Specification-Version": "25.0", |                     "Specification-Version": "1", | ||||||
|                     "Implementation-Title": "CC: Tweaked", |                     "Implementation-Title": "CC: Tweaked", | ||||||
|                     "Implementation-Version": "${mod_version}", |                     "Implementation-Version": "${mod_version}", | ||||||
|                     "Implementation-Vendor" :"SquidDev", |                     "Implementation-Vendor" :"SquidDev", | ||||||
| @@ -137,12 +134,22 @@ jar { | |||||||
|     from configurations.shade.collect { it.isDirectory() ? it : zipTree(it) } |     from configurations.shade.collect { it.isDirectory() ? it : zipTree(it) } | ||||||
| } | } | ||||||
|  |  | ||||||
|  | [compileJava, compileTestJava].forEach { | ||||||
|  |     it.configure { | ||||||
|  |         options.compilerArgs << "-Xlint" << "-Xlint:-processing" << "-Werror" | ||||||
|  |     } | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  |  | ||||||
| import java.nio.charset.StandardCharsets | import java.nio.charset.StandardCharsets | ||||||
| import java.nio.file.* | import java.nio.file.* | ||||||
| import java.util.zip.* | import java.util.zip.* | ||||||
|  |  | ||||||
| import com.google.gson.GsonBuilder | import com.google.gson.GsonBuilder | ||||||
| import com.google.gson.JsonElement | import com.google.gson.JsonElement | ||||||
|  | import com.hierynomus.gradle.license.tasks.LicenseCheck | ||||||
|  | import com.hierynomus.gradle.license.tasks.LicenseFormat | ||||||
| import org.ajoberstar.grgit.Grgit | import org.ajoberstar.grgit.Grgit | ||||||
| import proguard.gradle.ProGuardTask | import proguard.gradle.ProGuardTask | ||||||
|  |  | ||||||
| @@ -242,6 +249,7 @@ task compressJson(dependsOn: jar) { | |||||||
|         // Copy over all files in the current jar to the new one, running json files from GSON. As pretty printing |         // Copy over all files in the current jar to the new one, running json files from GSON. As pretty printing | ||||||
|         // is turned off, they should be minified. |         // is turned off, they should be minified. | ||||||
|         new ZipFile(jarPath).withCloseable { inJar -> |         new ZipFile(jarPath).withCloseable { inJar -> | ||||||
|  |             tempPath.getParentFile().mkdirs() | ||||||
|             new ZipOutputStream(new BufferedOutputStream(new FileOutputStream(tempPath))).withCloseable { outJar -> |             new ZipOutputStream(new BufferedOutputStream(new FileOutputStream(tempPath))).withCloseable { outJar -> | ||||||
|                 inJar.entries().each { entry -> |                 inJar.entries().each { entry -> | ||||||
|                     if(entry.directory) { |                     if(entry.directory) { | ||||||
| @@ -269,6 +277,56 @@ task compressJson(dependsOn: jar) { | |||||||
|  |  | ||||||
| assemble.dependsOn compressJson | assemble.dependsOn compressJson | ||||||
|  |  | ||||||
|  | // Check tasks | ||||||
|  |  | ||||||
|  | test { | ||||||
|  |     useJUnitPlatform() | ||||||
|  |     testLogging { | ||||||
|  |         events "skipped", "failed" | ||||||
|  |     } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | license { | ||||||
|  |     mapping("java", "SLASHSTAR_STYLE") | ||||||
|  |     strictCheck true | ||||||
|  |  | ||||||
|  |     ext.year = Calendar.getInstance().get(Calendar.YEAR) | ||||||
|  | } | ||||||
|  |  | ||||||
|  | [licenseMain, licenseFormatMain].forEach { | ||||||
|  |     it.configure { | ||||||
|  |         include("**/*.java") | ||||||
|  |         exclude("dan200/computercraft/api/**") | ||||||
|  |         header rootProject.file('config/license/main.txt') | ||||||
|  |     } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | [licenseTest, licenseFormatTest].forEach { | ||||||
|  |     it.configure { | ||||||
|  |         include("**/*.java") | ||||||
|  |         header rootProject.file('config/license/main.txt') | ||||||
|  |     } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | gradle.projectsEvaluated { | ||||||
|  |     tasks.withType(LicenseFormat) { | ||||||
|  |         outputs.upToDateWhen { false } | ||||||
|  |     } | ||||||
|  | } | ||||||
|  |  | ||||||
|  |  | ||||||
|  | task licenseAPI(type: LicenseCheck); | ||||||
|  | task licenseFormatAPI(type: LicenseFormat); | ||||||
|  | [licenseAPI, licenseFormatAPI].forEach { | ||||||
|  |     it.configure { | ||||||
|  |         source = sourceSets.main.java | ||||||
|  |         include("dan200/computercraft/api/**") | ||||||
|  |         header rootProject.file('config/license/api.txt') | ||||||
|  |     } | ||||||
|  | } | ||||||
|  |  | ||||||
|  | // Upload tasks | ||||||
|  |  | ||||||
| task checkRelease { | task checkRelease { | ||||||
|     group "upload" |     group "upload" | ||||||
|     description "Verifies that everything is ready for a release" |     description "Verifies that everything is ready for a release" | ||||||
| @@ -307,7 +365,7 @@ task checkRelease { | |||||||
|         if (!ok) throw new IllegalStateException("Could not check release") |         if (!ok) throw new IllegalStateException("Could not check release") | ||||||
|     } |     } | ||||||
| } | } | ||||||
|  | check.dependsOn checkRelease | ||||||
|  |  | ||||||
| curseforge { | curseforge { | ||||||
|     apiKey = project.hasProperty('curseForgeApiKey') ? project.curseForgeApiKey : '' |     apiKey = project.hasProperty('curseForgeApiKey') ? project.curseForgeApiKey : '' | ||||||
| @@ -379,7 +437,9 @@ githubRelease { | |||||||
|     token project.hasProperty('githubApiKey') ? project.githubApiKey : '' |     token project.hasProperty('githubApiKey') ? project.githubApiKey : '' | ||||||
|     owner 'SquidDev-CC' |     owner 'SquidDev-CC' | ||||||
|     repo 'CC-Tweaked' |     repo 'CC-Tweaked' | ||||||
|     targetCommitish { Grgit.open(dir: '.').branch.current().name } |     try { | ||||||
|  |         targetCommitish = Grgit.open(dir: '.').branch.current().name | ||||||
|  |     } catch(Exception ignored) { } | ||||||
|  |  | ||||||
|     tagName "v${mc_version}-${mod_version}" |     tagName "v${mc_version}-${mod_version}" | ||||||
|     releaseName "[${mc_version}] ${mod_version}" |     releaseName "[${mc_version}] ${mod_version}" | ||||||
| @@ -399,19 +459,3 @@ task uploadAll(dependsOn: uploadTasks) { | |||||||
|     group "upload" |     group "upload" | ||||||
|     description "Uploads to all repositories (Maven, Curse, GitHub release)" |     description "Uploads to all repositories (Maven, Curse, GitHub release)" | ||||||
| } | } | ||||||
|  |  | ||||||
| test { |  | ||||||
|     useJUnitPlatform() |  | ||||||
|     testLogging { |  | ||||||
|         events "passed", "skipped", "failed" |  | ||||||
|     } |  | ||||||
| } |  | ||||||
|  |  | ||||||
| gradle.projectsEvaluated { |  | ||||||
|     reobfJar.dependsOn proguardMove |  | ||||||
|  |  | ||||||
|     tasks.withType(JavaCompile) { |  | ||||||
|         options.compilerArgs << "-Xlint" << "-Xlint:-processing" // Causes Forge build to fail << "-Werror" |  | ||||||
|     } |  | ||||||
| } |  | ||||||
|  |  | ||||||
|   | |||||||
							
								
								
									
										169
									
								
								config/checkstyle/checkstyle.xml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										169
									
								
								config/checkstyle/checkstyle.xml
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,169 @@ | |||||||
|  | <?xml version="1.0" encoding="UTF-8"?> | ||||||
|  | <!DOCTYPE module PUBLIC | ||||||
|  |     "-//Checkstyle//DTD Checkstyle Configuration 1.3//EN" | ||||||
|  |     "https://checkstyle.org/dtds/configuration_1_3.dtd"> | ||||||
|  | <module name="Checker"> | ||||||
|  |     <property name="tabWidth" value="4"/> | ||||||
|  |     <property name="charset" value="UTF-8" /> | ||||||
|  |  | ||||||
|  |     <module name="SuppressionFilter"> | ||||||
|  |         <property name="file" value="config/checkstyle/suppressions.xml" /> | ||||||
|  |     </module> | ||||||
|  |  | ||||||
|  |     <module name="TreeWalker"> | ||||||
|  |         <!-- Annotations --> | ||||||
|  |         <module name="AnnotationLocation" /> | ||||||
|  |         <module name="AnnotationUseStyle" /> | ||||||
|  |         <module name="MissingDeprecated" /> | ||||||
|  |         <module name="MissingOverride" /> | ||||||
|  |  | ||||||
|  |         <!-- Blocks --> | ||||||
|  |         <module name="EmptyBlock" /> | ||||||
|  |         <module name="EmptyCatchBlock"> | ||||||
|  |             <property name="exceptionVariableName" value="ignored" /> | ||||||
|  |         </module> | ||||||
|  |         <module name="LeftCurly"> | ||||||
|  |             <property name="option" value="nl" /> | ||||||
|  |             <!-- The defaults, minus lambdas. --> | ||||||
|  |             <property name="tokens" value="ANNOTATION_DEF,CLASS_DEF,CTOR_DEF,ENUM_CONSTANT_DEF,ENUM_DEF,INTERFACE_DEF,LITERAL_CASE,LITERAL_CATCH,LITERAL_DEFAULT,LITERAL_DO,LITERAL_ELSE,LITERAL_FINALLY,LITERAL_FOR,LITERAL_IF,LITERAL_SWITCH,LITERAL_SYNCHRONIZED,LITERAL_TRY,LITERAL_WHILE,METHOD_DEF,OBJBLOCK,STATIC_INIT" /> | ||||||
|  |         </module> | ||||||
|  |         <module name="NeedBraces"> | ||||||
|  |             <property name="allowSingleLineStatement" value="true"/> | ||||||
|  |         </module> | ||||||
|  |         <module name="RightCurly"> | ||||||
|  |             <property name="option" value="alone" /> | ||||||
|  |         </module> | ||||||
|  |  | ||||||
|  |         <!-- Class design. As if we've ever followed good practice here. --> | ||||||
|  |         <module name="FinalClass" /> | ||||||
|  |         <module name="InterfaceIsType" /> | ||||||
|  |         <module name="MutableException" /> | ||||||
|  |         <module name="OneTopLevelClass" /> | ||||||
|  |  | ||||||
|  |         <!-- Coding --> | ||||||
|  |         <module name="ArrayTrailingComma" /> | ||||||
|  |         <module name="EqualsHashCode" /> | ||||||
|  |         <!-- FallThrough does not handle unreachable code well --> | ||||||
|  |         <module name="IllegalInstantiation" /> | ||||||
|  |         <module name="IllegalThrows" /> | ||||||
|  |         <module name="ModifiedControlVariable" /> | ||||||
|  |         <module name="NoClone" /> | ||||||
|  |         <module name="NoFinalizer" /> | ||||||
|  |         <module name="OneStatementPerLine" /> | ||||||
|  |         <module name="PackageDeclaration" /> | ||||||
|  |         <module name="SimplifyBooleanExpression" /> | ||||||
|  |         <module name="SimplifyBooleanReturn" /> | ||||||
|  |         <module name="StringLiteralEquality" /> | ||||||
|  |         <module name="UnnecessaryParentheses" /> | ||||||
|  |         <module name="UnnecessarySemicolonAfterTypeMemberDeclaration" /> | ||||||
|  |         <module name="UnnecessarySemicolonInTryWithResources" /> | ||||||
|  |         <module name="UnnecessarySemicolonInEnumeration" /> | ||||||
|  |  | ||||||
|  |         <!-- Imports --> | ||||||
|  |         <module name="CustomImportOrder" /> | ||||||
|  |         <module name="IllegalImport" /> | ||||||
|  |         <module name="RedundantImport" /> | ||||||
|  |         <module name="UnusedImports" /> | ||||||
|  |  | ||||||
|  |         <!-- Javadoc --> | ||||||
|  |         <!-- TODO: Missing* checks for the dan200.computercraft.api package? --> | ||||||
|  |         <module name="AtclauseOrder" /> | ||||||
|  |         <module name="InvalidJavadocPosition" /> | ||||||
|  |         <module name="JavadocBlockTagLocation" /> | ||||||
|  |         <module name="JavadocMethod"/> | ||||||
|  |         <module name="JavadocType"/> | ||||||
|  |         <module name="JavadocStyle" /> | ||||||
|  |         <module name="NonEmptyAtclauseDescription" /> | ||||||
|  |         <module name="SingleLineJavadoc" /> | ||||||
|  |         <module name="SummaryJavadocCheck"/> | ||||||
|  |  | ||||||
|  |         <!-- Misc --> | ||||||
|  |         <module name="ArrayTypeStyle" /> | ||||||
|  |         <module name="CommentsIndentation" /> | ||||||
|  |         <module name="Indentation" /> | ||||||
|  |         <module name="OuterTypeFilename" /> | ||||||
|  |  | ||||||
|  |         <!-- Modifiers --> | ||||||
|  |         <module name="ModifierOrder" /> | ||||||
|  |         <module name="RedundantModifier" /> | ||||||
|  |  | ||||||
|  |         <!-- Naming --> | ||||||
|  |         <module name="ClassTypeParameterName" /> | ||||||
|  |         <module name="InterfaceTypeParameterName" /> | ||||||
|  |         <module name="LambdaParameterName" /> | ||||||
|  |         <module name="LocalFinalVariableName" /> | ||||||
|  |         <module name="LocalVariableName" /> | ||||||
|  |         <!-- Allow an optional m_ on private members --> | ||||||
|  |         <module name="MemberName"> | ||||||
|  |             <property name="applyToPrivate" value="false" /> | ||||||
|  |             <property name="applyToPackage" value="false" /> | ||||||
|  |         </module> | ||||||
|  |         <module name="MemberName"> | ||||||
|  |             <property name="format" value="^(m_)?[a-z][a-zA-Z0-9]*$" /> | ||||||
|  |             <property name="applyToPrivate" value="true" /> | ||||||
|  |             <property name="applyToPackage" value="true" /> | ||||||
|  |         </module> | ||||||
|  |         <module name="MethodName" /> | ||||||
|  |         <module name="MethodTypeParameterName" /> | ||||||
|  |         <module name="PackageName"> | ||||||
|  |             <property name="format" value="^dan200\.computercraf(\.[a-z][a-z0-9]*)*" /> | ||||||
|  |         </module> | ||||||
|  |         <module name="ParameterName" /> | ||||||
|  |         <module name="StaticVariableName"> | ||||||
|  |             <property name="format" value="^[a-z][a-zA-Z0-9]*|CAPABILITY(_[A-Z]+)?$" /> | ||||||
|  |             <property name="applyToPrivate" value="false" /> | ||||||
|  |         </module> | ||||||
|  |         <module name="StaticVariableName"> | ||||||
|  |             <property name="format" value="^(s_)?[a-z][a-zA-Z0-9]*|CAPABILITY(_[A-Z]+)?$" /> | ||||||
|  |             <property name="applyToPrivate" value="true" /> | ||||||
|  |         </module> | ||||||
|  |         <module name="TypeName" /> | ||||||
|  |  | ||||||
|  |         <!-- Whitespace --> | ||||||
|  |         <module name="EmptyForInitializerPad"/> | ||||||
|  |         <module name="EmptyForIteratorPad"> | ||||||
|  |             <property name="option" value="space"/> | ||||||
|  |         </module> | ||||||
|  |         <module name="GenericWhitespace" /> | ||||||
|  |         <module name="MethodParamPad" /> | ||||||
|  |         <module name="NoLineWrap" /> | ||||||
|  |         <module name="NoWhitespaceAfter"> | ||||||
|  |             <property name="tokens" value="AT,INC,DEC,UNARY_MINUS,UNARY_PLUS,BNOT,LNOT,DOT,ARRAY_DECLARATOR,INDEX_OP" /> | ||||||
|  |         </module> | ||||||
|  |         <module name="NoWhitespaceBefore" /> | ||||||
|  |         <!-- TODO: Decide on an OperatorWrap style. --> | ||||||
|  |         <module name="ParenPad"> | ||||||
|  |             <property name="option" value="space" /> | ||||||
|  |             <property name="tokens" value="ANNOTATION,ANNOTATION_FIELD_DEF,CTOR_CALL,CTOR_DEF,ENUM_CONSTANT_DEF,LITERAL_CATCH,LITERAL_DO,LITERAL_FOR,LITERAL_IF,LITERAL_NEW,LITERAL_SWITCH,LITERAL_SYNCHRONIZED,LITERAL_WHILE,METHOD_CALL,METHOD_DEF,RESOURCE_SPECIFICATION,SUPER_CTOR_CALL,LAMBDA" /> | ||||||
|  |         </module> | ||||||
|  |         <module name="ParenPad"> | ||||||
|  |             <property name="option" value="nospace" /> | ||||||
|  |             <property name="tokens" value="DOT,EXPR,QUESTION" /> | ||||||
|  |         </module> | ||||||
|  |         <module name="SeparatorWrap"> | ||||||
|  |             <property name="option" value="eol" /> | ||||||
|  |             <property name="tokens" value="COMMA,SEMI,ELLIPSIS,ARRAY_DECLARATOR,RBRACK,METHOD_REF" /> | ||||||
|  |         </module> | ||||||
|  |         <module name="SeparatorWrap"> | ||||||
|  |             <property name="option" value="nl" /> | ||||||
|  |             <property name="tokens" value="DOT,AT" /> | ||||||
|  |         </module> | ||||||
|  |         <module name="SingleSpaceSeparator" /> | ||||||
|  |         <module name="TypecastParenPad" /> | ||||||
|  |         <module name="WhitespaceAfter"> | ||||||
|  |             <property name="tokens" value="COMMA" /> | ||||||
|  |         </module> | ||||||
|  |         <module name="WhitespaceAround"> | ||||||
|  |             <property name="allowEmptyConstructors" value="true" /> | ||||||
|  |             <property name="ignoreEnhancedForColon" value="false" /> | ||||||
|  |             <property name="tokens" value="ASSIGN,BAND,BAND_ASSIGN,BOR,BOR_ASSIGN,BSR,BSR_ASSIGN,BXOR,BXOR_ASSIGN,COLON,DIV,DIV_ASSIGN,EQUAL,GE,GT,LAMBDA,LAND,LCURLY,LE,LITERAL_RETURN,LOR,LT,MINUS,MINUS_ASSIGN,MOD,MOD_ASSIGN,NOT_EQUAL,PLUS,PLUS_ASSIGN,QUESTION,RCURLY,SL,SLIST,SL_ASSIGN,SR,SR_ASSIGN,STAR,STAR_ASSIGN,LITERAL_ASSERT,TYPE_EXTENSION_AND" /> | ||||||
|  |         </module> | ||||||
|  |     </module> | ||||||
|  |  | ||||||
|  |     <module name="FileTabCharacter" /> | ||||||
|  |     <module name="NewlineAtEndOfFile" /> | ||||||
|  |     <module name="RegexpSingleline"> | ||||||
|  |         <property name="format" value="\s+$"/> | ||||||
|  |         <property name="message" value="Trailing whitespace"/> | ||||||
|  |     </module> | ||||||
|  | </module> | ||||||
							
								
								
									
										12
									
								
								config/checkstyle/suppressions.xml
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										12
									
								
								config/checkstyle/suppressions.xml
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,12 @@ | |||||||
|  | <?xml version="1.0" encoding="UTF-8"?> | ||||||
|  | <!DOCTYPE suppressions PUBLIC | ||||||
|  |     "-//Checkstyle//DTD SuppressionFilter Configuration 1.2//EN" | ||||||
|  |     "https://checkstyle.org/dtds/suppressions_1_2.dtd"> | ||||||
|  | <suppressions> | ||||||
|  |     <!-- All the config options and method fields. --> | ||||||
|  |     <suppress checks="StaticVariableName" files=".*[\\/]ComputerCraft.java" /> | ||||||
|  |     <suppress checks="StaticVariableName" files=".*[\\/]ComputerCraftAPI.java" /> | ||||||
|  |  | ||||||
|  |     <!-- Do not check for missing package Javadoc. --> | ||||||
|  |     <suppress checks="JavadocStyle" files=".*[\\/]package-info.java" /> | ||||||
|  | </suppressions> | ||||||
							
								
								
									
										3
									
								
								config/license/api.txt
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										3
									
								
								config/license/api.txt
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,3 @@ | |||||||
|  | This file is part of the public ComputerCraft API - http://www.computercraft.info | ||||||
|  | Copyright Daniel Ratcliffe, 2011-${year}. 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. | ||||||
							
								
								
									
										3
									
								
								config/license/main.txt
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										3
									
								
								config/license/main.txt
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,3 @@ | |||||||
|  | This file is part of ComputerCraft - http://www.computercraft.info | ||||||
|  | Copyright Daniel Ratcliffe, 2011-${year}. Do not distribute without permission. | ||||||
|  | Send enquiries to dratcliffe@gmail.com | ||||||
| @@ -1,7 +1,7 @@ | |||||||
| # Mod properties | # Mod properties | ||||||
| mod_version=1.83.1 | mod_version=1.85.2 | ||||||
|  |  | ||||||
| # Minecraft properties | # Minecraft properties (update mods.toml when changing) | ||||||
| mc_version=1.13.2 | mc_version=1.14.4 | ||||||
| forge_version=25.0.219 | forge_version=28.1.71 | ||||||
| mappings_version=20190530-1.13.2 | mappings_version=20191123-1.14.3 | ||||||
|   | |||||||
| @@ -6,11 +6,9 @@ | |||||||
|  |  | ||||||
| package dan200.computercraft; | package dan200.computercraft; | ||||||
|  |  | ||||||
| import dan200.computercraft.api.filesystem.IMount; |  | ||||||
| import dan200.computercraft.api.turtle.event.TurtleAction; | import dan200.computercraft.api.turtle.event.TurtleAction; | ||||||
| import dan200.computercraft.core.apis.AddressPredicate; | import dan200.computercraft.core.apis.AddressPredicate; | ||||||
| import dan200.computercraft.core.apis.http.websocket.Websocket; | import dan200.computercraft.core.apis.http.websocket.Websocket; | ||||||
| import dan200.computercraft.core.filesystem.ResourceMount; |  | ||||||
| import dan200.computercraft.shared.Config; | import dan200.computercraft.shared.Config; | ||||||
| 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; | ||||||
| @@ -194,13 +192,6 @@ public final class ComputerCraft | |||||||
|         return "${version}"; |         return "${version}"; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     static IMount createResourceMount( String domain, String subPath ) |  | ||||||
|     { |  | ||||||
|         IReloadableResourceManager manager = ServerLifecycleHooks.getCurrentServer().getResourceManager(); |  | ||||||
|         ResourceMount mount = new ResourceMount( domain, subPath, manager ); |  | ||||||
|         return mount.exists( "" ) ? mount : null; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     public static InputStream getResourceFile( String domain, String subPath ) |     public static InputStream getResourceFile( String domain, String subPath ) | ||||||
|     { |     { | ||||||
|         IReloadableResourceManager manager = ServerLifecycleHooks.getCurrentServer().getResourceManager(); |         IReloadableResourceManager manager = ServerLifecycleHooks.getCurrentServer().getResourceManager(); | ||||||
|   | |||||||
| @@ -6,6 +6,7 @@ | |||||||
|  |  | ||||||
| package dan200.computercraft; | package dan200.computercraft; | ||||||
|  |  | ||||||
|  | import com.google.common.collect.MapMaker; | ||||||
| import dan200.computercraft.api.ComputerCraftAPI.IComputerCraftAPI; | import dan200.computercraft.api.ComputerCraftAPI.IComputerCraftAPI; | ||||||
| import dan200.computercraft.api.filesystem.IMount; | import dan200.computercraft.api.filesystem.IMount; | ||||||
| import dan200.computercraft.api.filesystem.IWritableMount; | import dan200.computercraft.api.filesystem.IWritableMount; | ||||||
| @@ -20,20 +21,26 @@ import dan200.computercraft.api.redstone.IBundledRedstoneProvider; | |||||||
| import dan200.computercraft.api.turtle.ITurtleUpgrade; | import dan200.computercraft.api.turtle.ITurtleUpgrade; | ||||||
| import dan200.computercraft.core.apis.ApiFactories; | import dan200.computercraft.core.apis.ApiFactories; | ||||||
| import dan200.computercraft.core.filesystem.FileMount; | import dan200.computercraft.core.filesystem.FileMount; | ||||||
|  | import dan200.computercraft.core.filesystem.ResourceMount; | ||||||
| import dan200.computercraft.shared.*; | import dan200.computercraft.shared.*; | ||||||
| import dan200.computercraft.shared.peripheral.modem.wireless.WirelessNetwork; | import dan200.computercraft.shared.peripheral.modem.wireless.WirelessNetwork; | ||||||
| import dan200.computercraft.shared.util.IDAssigner; | import dan200.computercraft.shared.util.IDAssigner; | ||||||
| import dan200.computercraft.shared.wired.CapabilityWiredElement; | import dan200.computercraft.shared.wired.CapabilityWiredElement; | ||||||
| import dan200.computercraft.shared.wired.WiredNode; | import dan200.computercraft.shared.wired.WiredNode; | ||||||
|  | import net.minecraft.resources.IReloadableResourceManager; | ||||||
| import net.minecraft.tileentity.TileEntity; | import net.minecraft.tileentity.TileEntity; | ||||||
| import net.minecraft.util.EnumFacing; | import net.minecraft.util.Direction; | ||||||
|  | import net.minecraft.util.ResourceLocation; | ||||||
| import net.minecraft.util.math.BlockPos; | import net.minecraft.util.math.BlockPos; | ||||||
| import net.minecraft.world.IBlockReader; | import net.minecraft.world.IBlockReader; | ||||||
| import net.minecraft.world.World; | import net.minecraft.world.World; | ||||||
| import net.minecraftforge.common.util.LazyOptional; | import net.minecraftforge.common.util.LazyOptional; | ||||||
|  | import net.minecraftforge.fml.server.ServerLifecycleHooks; | ||||||
|  |  | ||||||
| import javax.annotation.Nonnull; | import javax.annotation.Nonnull; | ||||||
| import java.io.File; | import java.io.File; | ||||||
|  | import java.lang.ref.WeakReference; | ||||||
|  | import java.util.Map; | ||||||
|  |  | ||||||
| public final class ComputerCraftAPIImpl implements IComputerCraftAPI | public final class ComputerCraftAPIImpl implements IComputerCraftAPI | ||||||
| { | { | ||||||
| @@ -43,6 +50,9 @@ public final class ComputerCraftAPIImpl implements IComputerCraftAPI | |||||||
|     { |     { | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     private WeakReference<IReloadableResourceManager> currentResources; | ||||||
|  |     private final Map<ResourceLocation, ResourceMount> mountCache = new MapMaker().weakValues().concurrencyLevel( 1 ).makeMap(); | ||||||
|  |  | ||||||
|     @Nonnull |     @Nonnull | ||||||
|     @Override |     @Override | ||||||
|     public String getInstalledVersion() |     public String getInstalledVersion() | ||||||
| @@ -72,7 +82,9 @@ public final class ComputerCraftAPIImpl implements IComputerCraftAPI | |||||||
|     @Override |     @Override | ||||||
|     public IMount createResourceMount( @Nonnull String domain, @Nonnull String subPath ) |     public IMount createResourceMount( @Nonnull String domain, @Nonnull String subPath ) | ||||||
|     { |     { | ||||||
|         return ComputerCraft.createResourceMount( domain, subPath ); |         IReloadableResourceManager manager = ServerLifecycleHooks.getCurrentServer().getResourceManager(); | ||||||
|  |         ResourceMount mount = ResourceMount.get( domain, subPath, manager ); | ||||||
|  |         return mount.exists( "" ) ? mount : null; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @Override |     @Override | ||||||
| @@ -94,7 +106,7 @@ public final class ComputerCraftAPIImpl implements IComputerCraftAPI | |||||||
|     } |     } | ||||||
|  |  | ||||||
|     @Override |     @Override | ||||||
|     public int getBundledRedstoneOutput( @Nonnull World world, @Nonnull BlockPos pos, @Nonnull EnumFacing side ) |     public int getBundledRedstoneOutput( @Nonnull World world, @Nonnull BlockPos pos, @Nonnull Direction side ) | ||||||
|     { |     { | ||||||
|         return BundledRedstone.getDefaultOutput( world, pos, side ); |         return BundledRedstone.getDefaultOutput( world, pos, side ); | ||||||
|     } |     } | ||||||
| @@ -133,7 +145,7 @@ public final class ComputerCraftAPIImpl implements IComputerCraftAPI | |||||||
|  |  | ||||||
|     @Nonnull |     @Nonnull | ||||||
|     @Override |     @Override | ||||||
|     public LazyOptional<IWiredElement> getWiredElementAt( @Nonnull IBlockReader world, @Nonnull BlockPos pos, @Nonnull EnumFacing side ) |     public LazyOptional<IWiredElement> getWiredElementAt( @Nonnull IBlockReader world, @Nonnull BlockPos pos, @Nonnull Direction side ) | ||||||
|     { |     { | ||||||
|         TileEntity tile = world.getTileEntity( pos ); |         TileEntity tile = world.getTileEntity( pos ); | ||||||
|         return tile == null ? LazyOptional.empty() : tile.getCapability( CapabilityWiredElement.CAPABILITY, side ); |         return tile == null ? LazyOptional.empty() : tile.getCapability( CapabilityWiredElement.CAPABILITY, side ); | ||||||
|   | |||||||
| @@ -1,9 +1,8 @@ | |||||||
| /* | /* | ||||||
|  * This file is part of ComputerCraft - http://www.computercraft.info |  * This file is part of the public ComputerCraft API - http://www.computercraft.info | ||||||
|  * Copyright Daniel Ratcliffe, 2011-2019. Do not distribute without permission. |  * Copyright Daniel Ratcliffe, 2011-2019. This API may be redistributed unmodified and in full only. | ||||||
|  * Send enquiries to dratcliffe@gmail.com |  * For help using the API, and posting your mods, visit the forums at computercraft.info. | ||||||
|  */ |  */ | ||||||
|  |  | ||||||
| package dan200.computercraft.api; | package dan200.computercraft.api; | ||||||
|  |  | ||||||
| import dan200.computercraft.api.turtle.ITurtleUpgrade; | import dan200.computercraft.api.turtle.ITurtleUpgrade; | ||||||
|   | |||||||
| @@ -20,7 +20,7 @@ import dan200.computercraft.api.peripheral.IPeripheralProvider; | |||||||
| 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.Direction; | ||||||
| import net.minecraft.util.math.BlockPos; | import net.minecraft.util.math.BlockPos; | ||||||
| import net.minecraft.world.IBlockReader; | import net.minecraft.world.IBlockReader; | ||||||
| import net.minecraft.world.World; | import net.minecraft.world.World; | ||||||
| @@ -31,8 +31,9 @@ import javax.annotation.Nullable; | |||||||
|  |  | ||||||
| /** | /** | ||||||
|  * The static entry point to the ComputerCraft API. |  * The static entry point to the ComputerCraft API. | ||||||
|  * Members in this class must be called after mod_ComputerCraft has been initialised, |  * | ||||||
|  * but may be called before it is fully loaded. |  * Members in this class must be called after mod_ComputerCraft has been initialised, but may be called before it is | ||||||
|  |  * fully loaded. | ||||||
|  */ |  */ | ||||||
| public final class ComputerCraftAPI | public final class ComputerCraftAPI | ||||||
| { | { | ||||||
| @@ -183,13 +184,13 @@ public final class ComputerCraftAPI | |||||||
|      * If there is no block capable of emitting bundled redstone at the location, -1 will be returned. |      * If there is no block capable of emitting bundled redstone at the location, -1 will be returned. | ||||||
|      * @see IBundledRedstoneProvider |      * @see IBundledRedstoneProvider | ||||||
|      */ |      */ | ||||||
|     public static int getBundledRedstoneOutput( @Nonnull World world, @Nonnull BlockPos pos, @Nonnull EnumFacing side ) |     public static int getBundledRedstoneOutput( @Nonnull World world, @Nonnull BlockPos pos, @Nonnull Direction side ) | ||||||
|     { |     { | ||||||
|         return getInstance().getBundledRedstoneOutput( world, pos, side ); |         return getInstance().getBundledRedstoneOutput( world, pos, side ); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * Registers a media provider to provide {@link IMedia} implementations for Items |      * Registers a media provider to provide {@link IMedia} implementations for Items. | ||||||
|      * |      * | ||||||
|      * @param provider The media provider to register. |      * @param provider The media provider to register. | ||||||
|      * @see IMediaProvider |      * @see IMediaProvider | ||||||
| @@ -220,7 +221,7 @@ public final class ComputerCraftAPI | |||||||
|     } |     } | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * Construct a new wired node for a given wired element |      * Construct a new wired node for a given wired element. | ||||||
|      * |      * | ||||||
|      * @param element The element to construct it for |      * @param element The element to construct it for | ||||||
|      * @return The element's node |      * @return The element's node | ||||||
| @@ -233,7 +234,7 @@ public final class ComputerCraftAPI | |||||||
|     } |     } | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * Get the wired network element for a block in world |      * Get the wired network element for a block in world. | ||||||
|      * |      * | ||||||
|      * @param world The world the block exists in |      * @param world The world the block exists in | ||||||
|      * @param pos   The position the block exists in |      * @param pos   The position the block exists in | ||||||
| @@ -242,7 +243,7 @@ public final class ComputerCraftAPI | |||||||
|      * @see IWiredElement#getNode() |      * @see IWiredElement#getNode() | ||||||
|      */ |      */ | ||||||
|     @Nonnull |     @Nonnull | ||||||
|     public static LazyOptional<IWiredElement> getWiredElementAt( @Nonnull IBlockReader world, @Nonnull BlockPos pos, @Nonnull EnumFacing side ) |     public static LazyOptional<IWiredElement> getWiredElementAt( @Nonnull IBlockReader world, @Nonnull BlockPos pos, @Nonnull Direction side ) | ||||||
|     { |     { | ||||||
|         return getInstance().getWiredElementAt( world, pos, side ); |         return getInstance().getWiredElementAt( world, pos, side ); | ||||||
|     } |     } | ||||||
| @@ -284,7 +285,7 @@ public final class ComputerCraftAPI | |||||||
|  |  | ||||||
|         void registerBundledRedstoneProvider( @Nonnull IBundledRedstoneProvider provider ); |         void registerBundledRedstoneProvider( @Nonnull IBundledRedstoneProvider provider ); | ||||||
|  |  | ||||||
|         int getBundledRedstoneOutput( @Nonnull World world, @Nonnull BlockPos pos, @Nonnull EnumFacing side ); |         int getBundledRedstoneOutput( @Nonnull World world, @Nonnull BlockPos pos, @Nonnull Direction side ); | ||||||
|  |  | ||||||
|         void registerMediaProvider( @Nonnull IMediaProvider provider ); |         void registerMediaProvider( @Nonnull IMediaProvider provider ); | ||||||
|  |  | ||||||
| @@ -299,6 +300,6 @@ public final class ComputerCraftAPI | |||||||
|         IWiredNode createWiredNodeForElement( @Nonnull IWiredElement element ); |         IWiredNode createWiredNodeForElement( @Nonnull IWiredElement element ); | ||||||
|  |  | ||||||
|         @Nonnull |         @Nonnull | ||||||
|         LazyOptional<IWiredElement> getWiredElementAt( @Nonnull IBlockReader world, @Nonnull BlockPos pos, @Nonnull EnumFacing side ); |         LazyOptional<IWiredElement> getWiredElementAt( @Nonnull IBlockReader world, @Nonnull BlockPos pos, @Nonnull Direction side ); | ||||||
|     } |     } | ||||||
| } | } | ||||||
|   | |||||||
| @@ -0,0 +1,41 @@ | |||||||
|  | /* | ||||||
|  |  * 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.filesystem; | ||||||
|  |  | ||||||
|  | import javax.annotation.Nonnull; | ||||||
|  | import javax.annotation.Nullable; | ||||||
|  | import java.io.IOException; | ||||||
|  | import java.util.Objects; | ||||||
|  |  | ||||||
|  | /** | ||||||
|  |  * An {@link IOException} which occurred on a specific file. | ||||||
|  |  * | ||||||
|  |  * This may be thrown from a {@link IMount} or {@link IWritableMount} to give more information about a failure. | ||||||
|  |  */ | ||||||
|  | public class FileOperationException extends IOException | ||||||
|  | { | ||||||
|  |     private static final long serialVersionUID = -8809108200853029849L; | ||||||
|  |  | ||||||
|  |     private final String filename; | ||||||
|  |  | ||||||
|  |     public FileOperationException( @Nullable String filename, @Nonnull String message ) | ||||||
|  |     { | ||||||
|  |         super( Objects.requireNonNull( message, "message cannot be null" ) ); | ||||||
|  |         this.filename = filename; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     public FileOperationException( String message ) | ||||||
|  |     { | ||||||
|  |         super( Objects.requireNonNull( message, "message cannot be null" ) ); | ||||||
|  |         this.filename = null; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     @Nullable | ||||||
|  |     public String getFilename() | ||||||
|  |     { | ||||||
|  |         return filename; | ||||||
|  |     } | ||||||
|  | } | ||||||
| @@ -19,7 +19,7 @@ import java.util.List; | |||||||
|  |  | ||||||
| /** | /** | ||||||
|  * Represents a read only part of a virtual filesystem that can be mounted onto a computer using |  * Represents a read only part of a virtual filesystem that can be mounted onto a computer using | ||||||
|  * {@link IComputerAccess#mount(String, IMount)} |  * {@link IComputerAccess#mount(String, IMount)}. | ||||||
|  * |  * | ||||||
|  * Ready made implementations of this interface can be created using |  * Ready made implementations of this interface can be created using | ||||||
|  * {@link ComputerCraftAPI#createSaveDirMount(World, String, long)} or |  * {@link ComputerCraftAPI#createSaveDirMount(World, String, long)} or | ||||||
| @@ -60,7 +60,7 @@ public interface IMount | |||||||
|     void list( @Nonnull String path, @Nonnull List<String> contents ) throws IOException; |     void list( @Nonnull String path, @Nonnull List<String> contents ) throws IOException; | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * Returns the size of a file with a given path, in bytes |      * Returns the size of a file with a given path, in bytes. | ||||||
|      * |      * | ||||||
|      * @param path A file path in normalised format, relative to the mount location. ie: "programs/myprogram". |      * @param path A file path in normalised format, relative to the mount location. ie: "programs/myprogram". | ||||||
|      * @return The size of the file, in bytes. |      * @return The size of the file, in bytes. | ||||||
| @@ -90,7 +90,6 @@ public interface IMount | |||||||
|      * @throws IOException If the file does not exist, or could not be opened. |      * @throws IOException If the file does not exist, or could not be opened. | ||||||
|      */ |      */ | ||||||
|     @Nonnull |     @Nonnull | ||||||
|     @SuppressWarnings( "deprecation" ) |  | ||||||
|     default ReadableByteChannel openChannelForRead( @Nonnull String path ) throws IOException |     default ReadableByteChannel openChannelForRead( @Nonnull String path ) throws IOException | ||||||
|     { |     { | ||||||
|         return Channels.newChannel( openForRead( path ) ); |         return Channels.newChannel( openForRead( path ) ); | ||||||
|   | |||||||
| @@ -67,7 +67,6 @@ public interface IWritableMount extends IMount | |||||||
|      * @throws IOException If the file could not be opened for writing. |      * @throws IOException If the file could not be opened for writing. | ||||||
|      */ |      */ | ||||||
|     @Nonnull |     @Nonnull | ||||||
|     @SuppressWarnings( "deprecation" ) |  | ||||||
|     default WritableByteChannel openChannelForWrite( @Nonnull String path ) throws IOException |     default WritableByteChannel openChannelForWrite( @Nonnull String path ) throws IOException | ||||||
|     { |     { | ||||||
|         return Channels.newChannel( openForWrite( path ) ); |         return Channels.newChannel( openForWrite( path ) ); | ||||||
| @@ -94,7 +93,6 @@ public interface IWritableMount extends IMount | |||||||
|      * @throws IOException If the file could not be opened for writing. |      * @throws IOException If the file could not be opened for writing. | ||||||
|      */ |      */ | ||||||
|     @Nonnull |     @Nonnull | ||||||
|     @SuppressWarnings( "deprecation" ) |  | ||||||
|     default WritableByteChannel openChannelForAppend( @Nonnull String path ) throws IOException |     default WritableByteChannel openChannelForAppend( @Nonnull String path ) throws IOException | ||||||
|     { |     { | ||||||
|         return Channels.newChannel( openForAppend( path ) ); |         return Channels.newChannel( openForAppend( path ) ); | ||||||
|   | |||||||
							
								
								
									
										335
									
								
								src/main/java/dan200/computercraft/api/lua/ArgumentHelper.java
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										335
									
								
								src/main/java/dan200/computercraft/api/lua/ArgumentHelper.java
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,335 @@ | |||||||
|  | /* | ||||||
|  |  * 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.lua; | ||||||
|  |  | ||||||
|  | import dan200.computercraft.api.peripheral.IComputerAccess; | ||||||
|  | import dan200.computercraft.api.peripheral.IPeripheral; | ||||||
|  |  | ||||||
|  | import javax.annotation.Nonnull; | ||||||
|  | import javax.annotation.Nullable; | ||||||
|  | import java.util.Map; | ||||||
|  |  | ||||||
|  | /** | ||||||
|  |  * Provides methods for extracting values and validating Lua arguments, such as those provided to | ||||||
|  |  * {@link ILuaObject#callMethod(ILuaContext, int, Object[])} or | ||||||
|  |  * {@link IPeripheral#callMethod(IComputerAccess, ILuaContext, int, Object[])}. | ||||||
|  |  * | ||||||
|  |  * This provides two sets of functions: the {@code get*} methods, which require an argument to be valid, and | ||||||
|  |  * {@code opt*}, which accept a default value and return that if the argument was not present or was {@code null}. | ||||||
|  |  * If the argument is of the wrong type, a suitable error message will be thrown, with a similar format to Lua's own | ||||||
|  |  * error messages. | ||||||
|  |  * | ||||||
|  |  * <h2>Example usage:</h2> | ||||||
|  |  * <pre> | ||||||
|  |  * {@code | ||||||
|  |  * int slot = getInt( args, 0 ); | ||||||
|  |  * int amount = optInt( args, 1, 64 ); | ||||||
|  |  * } | ||||||
|  |  * </pre> | ||||||
|  |  */ | ||||||
|  | public final class ArgumentHelper | ||||||
|  | { | ||||||
|  |     private ArgumentHelper() | ||||||
|  |     { | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * Get a string representation of the given value's type. | ||||||
|  |      * | ||||||
|  |      * @param value The value whose type we are trying to compute. | ||||||
|  |      * @return A string representation of the given value's type, in a similar format to that provided by Lua's | ||||||
|  |      * {@code type} function. | ||||||
|  |      */ | ||||||
|  |     @Nonnull | ||||||
|  |     public static String getType( @Nullable Object value ) | ||||||
|  |     { | ||||||
|  |         if( value == null ) return "nil"; | ||||||
|  |         if( value instanceof String ) return "string"; | ||||||
|  |         if( value instanceof Boolean ) return "boolean"; | ||||||
|  |         if( value instanceof Number ) return "number"; | ||||||
|  |         if( value instanceof Map ) return "table"; | ||||||
|  |         return "userdata"; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * Construct a "bad argument" exception, from an expected type and the actual value provided. | ||||||
|  |      * | ||||||
|  |      * @param index    The argument number, starting from 0. | ||||||
|  |      * @param expected The expected type for this argument. | ||||||
|  |      * @param actual   The actual value provided for this argument. | ||||||
|  |      * @return The constructed exception, which should be thrown immediately. | ||||||
|  |      */ | ||||||
|  |     @Nonnull | ||||||
|  |     public static LuaException badArgumentOf( int index, @Nonnull String expected, @Nullable Object actual ) | ||||||
|  |     { | ||||||
|  |         return badArgument( index, expected, getType( actual ) ); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * Construct a "bad argument" exception, from an expected and actual type. | ||||||
|  |      * | ||||||
|  |      * @param index    The argument number, starting from 0. | ||||||
|  |      * @param expected The expected type for this argument. | ||||||
|  |      * @param actual   The provided type for this argument. | ||||||
|  |      * @return The constructed exception, which should be thrown immediately. | ||||||
|  |      */ | ||||||
|  |     @Nonnull | ||||||
|  |     public static LuaException badArgument( int index, @Nonnull String expected, @Nonnull String actual ) | ||||||
|  |     { | ||||||
|  |         return new LuaException( "bad argument #" + (index + 1) + " (" + expected + " expected, got " + actual + ")" ); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * Get an argument as a double. | ||||||
|  |      * | ||||||
|  |      * @param args  The arguments to extract from. | ||||||
|  |      * @param index The index into the argument array to read from. | ||||||
|  |      * @return The argument's value. | ||||||
|  |      * @throws LuaException If the value is not a number. | ||||||
|  |      * @see #getFiniteDouble(Object[], int) if you require this to be finite (i.e. not infinite or NaN). | ||||||
|  |      */ | ||||||
|  |     public static double getDouble( @Nonnull Object[] args, int index ) throws LuaException | ||||||
|  |     { | ||||||
|  |         if( index >= args.length ) throw badArgument( index, "number", "nil" ); | ||||||
|  |         Object value = args[index]; | ||||||
|  |         if( !(value instanceof Number) ) throw badArgumentOf( index, "number", value ); | ||||||
|  |         return ((Number) value).doubleValue(); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * Get an argument as an integer. | ||||||
|  |      * | ||||||
|  |      * @param args  The arguments to extract from. | ||||||
|  |      * @param index The index into the argument array to read from. | ||||||
|  |      * @return The argument's value. | ||||||
|  |      * @throws LuaException If the value is not an integer. | ||||||
|  |      */ | ||||||
|  |     public static int getInt( @Nonnull Object[] args, int index ) throws LuaException | ||||||
|  |     { | ||||||
|  |         return (int) getLong( args, index ); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * Get an argument as a long. | ||||||
|  |      * | ||||||
|  |      * @param args  The arguments to extract from. | ||||||
|  |      * @param index The index into the argument array to read from. | ||||||
|  |      * @return The argument's value. | ||||||
|  |      * @throws LuaException If the value is not a long. | ||||||
|  |      */ | ||||||
|  |     public static long getLong( @Nonnull Object[] args, int index ) throws LuaException | ||||||
|  |     { | ||||||
|  |         if( index >= args.length ) throw badArgument( index, "number", "nil" ); | ||||||
|  |         Object value = args[index]; | ||||||
|  |         if( !(value instanceof Number) ) throw badArgumentOf( index, "number", value ); | ||||||
|  |         return checkFinite( index, (Number) value ).longValue(); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * Get an argument as a finite number (not infinite or NaN). | ||||||
|  |      * | ||||||
|  |      * @param args  The arguments to extract from. | ||||||
|  |      * @param index The index into the argument array to read from. | ||||||
|  |      * @return The argument's value. | ||||||
|  |      * @throws LuaException If the value is not finite. | ||||||
|  |      */ | ||||||
|  |     public static double getFiniteDouble( @Nonnull Object[] args, int index ) throws LuaException | ||||||
|  |     { | ||||||
|  |         return checkFinite( index, getDouble( args, index ) ); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * Get an argument as a boolean. | ||||||
|  |      * | ||||||
|  |      * @param args  The arguments to extract from. | ||||||
|  |      * @param index The index into the argument array to read from. | ||||||
|  |      * @return The argument's value. | ||||||
|  |      * @throws LuaException If the value is not a boolean. | ||||||
|  |      */ | ||||||
|  |     public static boolean getBoolean( @Nonnull Object[] args, int index ) throws LuaException | ||||||
|  |     { | ||||||
|  |         if( index >= args.length ) throw badArgument( index, "boolean", "nil" ); | ||||||
|  |         Object value = args[index]; | ||||||
|  |         if( !(value instanceof Boolean) ) throw badArgumentOf( index, "boolean", value ); | ||||||
|  |         return (Boolean) value; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * Get an argument as a string. | ||||||
|  |      * | ||||||
|  |      * @param args  The arguments to extract from. | ||||||
|  |      * @param index The index into the argument array to read from. | ||||||
|  |      * @return The argument's value. | ||||||
|  |      * @throws LuaException If the value is not a string. | ||||||
|  |      */ | ||||||
|  |     @Nonnull | ||||||
|  |     public static String getString( @Nonnull Object[] args, int index ) throws LuaException | ||||||
|  |     { | ||||||
|  |         if( index >= args.length ) throw badArgument( index, "string", "nil" ); | ||||||
|  |         Object value = args[index]; | ||||||
|  |         if( !(value instanceof String) ) throw badArgumentOf( index, "string", value ); | ||||||
|  |         return (String) value; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * Get an argument as a table. | ||||||
|  |      * | ||||||
|  |      * @param args  The arguments to extract from. | ||||||
|  |      * @param index The index into the argument array to read from. | ||||||
|  |      * @return The argument's value. | ||||||
|  |      * @throws LuaException If the value is not a table. | ||||||
|  |      */ | ||||||
|  |     @Nonnull | ||||||
|  |     public static Map<?, ?> getTable( @Nonnull Object[] args, int index ) throws LuaException | ||||||
|  |     { | ||||||
|  |         if( index >= args.length ) throw badArgument( index, "table", "nil" ); | ||||||
|  |         Object value = args[index]; | ||||||
|  |         if( !(value instanceof Map) ) throw badArgumentOf( index, "table", value ); | ||||||
|  |         return (Map<?, ?>) value; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * Get an argument as a double. | ||||||
|  |      * | ||||||
|  |      * @param args  The arguments to extract from. | ||||||
|  |      * @param index The index into the argument array to read from. | ||||||
|  |      * @param def   The default value, if this argument is not given. | ||||||
|  |      * @return The argument's value, or {@code def} if none was provided. | ||||||
|  |      * @throws LuaException If the value is not a number. | ||||||
|  |      */ | ||||||
|  |     public static double optDouble( @Nonnull Object[] args, int index, double def ) throws LuaException | ||||||
|  |     { | ||||||
|  |         Object value = index < args.length ? args[index] : null; | ||||||
|  |         if( value == null ) return def; | ||||||
|  |         if( !(value instanceof Number) ) throw badArgumentOf( index, "number", value ); | ||||||
|  |         return ((Number) value).doubleValue(); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * Get an argument as an int. | ||||||
|  |      * | ||||||
|  |      * @param args  The arguments to extract from. | ||||||
|  |      * @param index The index into the argument array to read from. | ||||||
|  |      * @param def   The default value, if this argument is not given. | ||||||
|  |      * @return The argument's value, or {@code def} if none was provided. | ||||||
|  |      * @throws LuaException If the value is not a number. | ||||||
|  |      */ | ||||||
|  |     public static int optInt( @Nonnull Object[] args, int index, int def ) throws LuaException | ||||||
|  |     { | ||||||
|  |         return (int) optLong( args, index, def ); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * Get an argument as a long. | ||||||
|  |      * | ||||||
|  |      * @param args  The arguments to extract from. | ||||||
|  |      * @param index The index into the argument array to read from. | ||||||
|  |      * @param def   The default value, if this argument is not given. | ||||||
|  |      * @return The argument's value, or {@code def} if none was provided. | ||||||
|  |      * @throws LuaException If the value is not a number. | ||||||
|  |      */ | ||||||
|  |     public static long optLong( @Nonnull Object[] args, int index, long def ) throws LuaException | ||||||
|  |     { | ||||||
|  |         Object value = index < args.length ? args[index] : null; | ||||||
|  |         if( value == null ) return def; | ||||||
|  |         if( !(value instanceof Number) ) throw badArgumentOf( index, "number", value ); | ||||||
|  |         return checkFinite( index, (Number) value ).longValue(); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * Get an argument as a finite number (not infinite or NaN). | ||||||
|  |      * | ||||||
|  |      * @param args  The arguments to extract from. | ||||||
|  |      * @param index The index into the argument array to read from. | ||||||
|  |      * @param def   The default value, if this argument is not given. | ||||||
|  |      * @return The argument's value, or {@code def} if none was provided. | ||||||
|  |      * @throws LuaException If the value is not finite. | ||||||
|  |      */ | ||||||
|  |     public static double optFiniteDouble( @Nonnull Object[] args, int index, double def ) throws LuaException | ||||||
|  |     { | ||||||
|  |         return checkFinite( index, optDouble( args, index, def ) ); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * Get an argument as a boolean. | ||||||
|  |      * | ||||||
|  |      * @param args  The arguments to extract from. | ||||||
|  |      * @param index The index into the argument array to read from. | ||||||
|  |      * @param def   The default value, if this argument is not given. | ||||||
|  |      * @return The argument's value, or {@code def} if none was provided. | ||||||
|  |      * @throws LuaException If the value is not a boolean. | ||||||
|  |      */ | ||||||
|  |     public static boolean optBoolean( @Nonnull Object[] args, int index, boolean def ) throws LuaException | ||||||
|  |     { | ||||||
|  |         Object value = index < args.length ? args[index] : null; | ||||||
|  |         if( value == null ) return def; | ||||||
|  |         if( !(value instanceof Boolean) ) throw badArgumentOf( index, "boolean", value ); | ||||||
|  |         return (Boolean) value; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * Get an argument as a string. | ||||||
|  |      * | ||||||
|  |      * @param args  The arguments to extract from. | ||||||
|  |      * @param index The index into the argument array to read from. | ||||||
|  |      * @param def   The default value, if this argument is not given. | ||||||
|  |      * @return The argument's value, or {@code def} if none was provided. | ||||||
|  |      * @throws LuaException If the value is not a string. | ||||||
|  |      */ | ||||||
|  |     public static String optString( @Nonnull Object[] args, int index, String def ) throws LuaException | ||||||
|  |     { | ||||||
|  |         Object value = index < args.length ? args[index] : null; | ||||||
|  |         if( value == null ) return def; | ||||||
|  |         if( !(value instanceof String) ) throw badArgumentOf( index, "string", value ); | ||||||
|  |         return (String) value; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * Get an argument as a table. | ||||||
|  |      * | ||||||
|  |      * @param args  The arguments to extract from. | ||||||
|  |      * @param index The index into the argument array to read from. | ||||||
|  |      * @param def   The default value, if this argument is not given. | ||||||
|  |      * @return The argument's value, or {@code def} if none was provided. | ||||||
|  |      * @throws LuaException If the value is not a table. | ||||||
|  |      */ | ||||||
|  |     public static Map<?, ?> optTable( @Nonnull Object[] args, int index, Map<Object, Object> def ) throws LuaException | ||||||
|  |     { | ||||||
|  |         Object value = index < args.length ? args[index] : null; | ||||||
|  |         if( value == null ) return def; | ||||||
|  |         if( !(value instanceof Map) ) throw badArgumentOf( index, "table", value ); | ||||||
|  |         return (Map<?, ?>) value; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     private static Number checkFinite( int index, Number value ) throws LuaException | ||||||
|  |     { | ||||||
|  |         checkFinite( index, value.doubleValue() ); | ||||||
|  |         return value; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     private static double checkFinite( int index, double value ) throws LuaException | ||||||
|  |     { | ||||||
|  |         if( !Double.isFinite( value ) ) throw badArgument( index, "number", getNumericType( value ) ); | ||||||
|  |         return value; | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * Returns a more detailed representation of this number's type. If this is finite, it will just return "number", | ||||||
|  |      * otherwise it returns whether it is infinite or NaN. | ||||||
|  |      * | ||||||
|  |      * @param value The value to extract the type for. | ||||||
|  |      * @return This value's numeric type. | ||||||
|  |      */ | ||||||
|  |     @Nonnull | ||||||
|  |     public static String getNumericType( double value ) | ||||||
|  |     { | ||||||
|  |         if( Double.isNaN( value ) ) return "nan"; | ||||||
|  |         if( value == Double.POSITIVE_INFINITY ) return "inf"; | ||||||
|  |         if( value == Double.NEGATIVE_INFINITY ) return "-inf"; | ||||||
|  |         return "number"; | ||||||
|  |     } | ||||||
|  | } | ||||||
| @@ -26,7 +26,7 @@ public interface IComputerSystem extends IComputerAccess | |||||||
|     IFileSystem getFileSystem(); |     IFileSystem getFileSystem(); | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * Get the label for this computer |      * Get the label for this computer. | ||||||
|      * |      * | ||||||
|      * @return This computer's label, or {@code null} if it is not set. |      * @return This computer's label, or {@code null} if it is not set. | ||||||
|      */ |      */ | ||||||
|   | |||||||
| @@ -6,6 +6,7 @@ | |||||||
|  |  | ||||||
| package dan200.computercraft.api.peripheral; | package dan200.computercraft.api.peripheral; | ||||||
|  |  | ||||||
|  | import dan200.computercraft.api.lua.ArgumentHelper; | ||||||
| import dan200.computercraft.api.lua.ILuaContext; | import dan200.computercraft.api.lua.ILuaContext; | ||||||
| import dan200.computercraft.api.lua.LuaException; | import dan200.computercraft.api.lua.LuaException; | ||||||
|  |  | ||||||
| @@ -58,9 +59,11 @@ public interface IPeripheral | |||||||
|      *                  Lua values of type "table" will be represented by Object type Map.<br> |      *                  Lua values of type "table" will be represented by Object type Map.<br> | ||||||
|      *                  Lua values of any other type will be represented by a null object.<br> |      *                  Lua values of any other type will be represented by a null object.<br> | ||||||
|      *                  This array will be empty if no arguments are passed. |      *                  This array will be empty if no arguments are passed. | ||||||
|  |      * | ||||||
|  |      *                  It is recommended you use {@link ArgumentHelper} in order to validate and process arguments. | ||||||
|      * @return An array of objects, representing values you wish to return to the lua program. Integers, Doubles, Floats, |      * @return An array of objects, representing values you wish to return to the lua program. Integers, Doubles, Floats, | ||||||
|      * Strings, Booleans, Maps and ILuaObject and null be converted to their corresponding lua type. All other types |      * Strings, Booleans, Maps, ILuaObject and null be converted to their corresponding lua type. All other types will | ||||||
|      * will be converted to nil. |      * be converted to nil. | ||||||
|      * |      * | ||||||
|      * You may return null to indicate no values should be returned. |      * You may return null to indicate no values should be returned. | ||||||
|      * @throws LuaException         If you throw any exception from this function, a lua error will be raised with the |      * @throws LuaException         If you throw any exception from this function, a lua error will be raised with the | ||||||
| @@ -70,6 +73,7 @@ public interface IPeripheral | |||||||
|      *                              InterruptedException will be thrown. This exception must not be caught or |      *                              InterruptedException will be thrown. This exception must not be caught or | ||||||
|      *                              intercepted, or the computer will leak memory and end up in a broken state. |      *                              intercepted, or the computer will leak memory and end up in a broken state. | ||||||
|      * @see #getMethodNames |      * @see #getMethodNames | ||||||
|  |      * @see ArgumentHelper | ||||||
|      */ |      */ | ||||||
|     @Nullable |     @Nullable | ||||||
|     Object[] callMethod( @Nonnull IComputerAccess computer, @Nonnull ILuaContext context, int method, @Nonnull Object[] arguments ) throws LuaException, InterruptedException; |     Object[] callMethod( @Nonnull IComputerAccess computer, @Nonnull ILuaContext context, int method, @Nonnull Object[] arguments ) throws LuaException, InterruptedException; | ||||||
|   | |||||||
| @@ -7,7 +7,7 @@ | |||||||
| package dan200.computercraft.api.peripheral; | package dan200.computercraft.api.peripheral; | ||||||
|  |  | ||||||
| import net.minecraft.tileentity.TileEntity; | import net.minecraft.tileentity.TileEntity; | ||||||
| import net.minecraft.util.EnumFacing; | import net.minecraft.util.Direction; | ||||||
| import net.minecraft.util.math.BlockPos; | import net.minecraft.util.math.BlockPos; | ||||||
| import net.minecraft.world.World; | import net.minecraft.world.World; | ||||||
|  |  | ||||||
| @@ -34,5 +34,5 @@ public interface IPeripheralProvider | |||||||
|      * @see dan200.computercraft.api.ComputerCraftAPI#registerPeripheralProvider(IPeripheralProvider) |      * @see dan200.computercraft.api.ComputerCraftAPI#registerPeripheralProvider(IPeripheralProvider) | ||||||
|      */ |      */ | ||||||
|     @Nullable |     @Nullable | ||||||
|     IPeripheral getPeripheral( @Nonnull World world, @Nonnull BlockPos pos, @Nonnull EnumFacing side ); |     IPeripheral getPeripheral( @Nonnull World world, @Nonnull BlockPos pos, @Nonnull Direction side ); | ||||||
| } | } | ||||||
|   | |||||||
| @@ -5,7 +5,7 @@ | |||||||
|  */ |  */ | ||||||
| package dan200.computercraft.api.peripheral; | package dan200.computercraft.api.peripheral; | ||||||
|  |  | ||||||
| import net.minecraft.util.EnumFacing; | import net.minecraft.util.Direction; | ||||||
| import net.minecraft.util.math.BlockPos; | import net.minecraft.util.math.BlockPos; | ||||||
| import net.minecraft.world.World; | import net.minecraft.world.World; | ||||||
|  |  | ||||||
| @@ -25,8 +25,8 @@ public interface IPeripheralTile | |||||||
|      * |      * | ||||||
|      * @param side The side to get the peripheral from. |      * @param side The side to get the peripheral from. | ||||||
|      * @return A peripheral, or {@code null} if there is not a peripheral here. |      * @return A peripheral, or {@code null} if there is not a peripheral here. | ||||||
|      * @see IPeripheralProvider#getPeripheral(World, BlockPos, EnumFacing) |      * @see IPeripheralProvider#getPeripheral(World, BlockPos, Direction) | ||||||
|      */ |      */ | ||||||
|     @Nullable |     @Nullable | ||||||
|     IPeripheral getPeripheral( @Nonnull EnumFacing side ); |     IPeripheral getPeripheral( @Nonnull Direction side ); | ||||||
| } | } | ||||||
|   | |||||||
| @@ -1,9 +1,8 @@ | |||||||
| /* | /* | ||||||
|  * This file is part of ComputerCraft - http://www.computercraft.info |  * This file is part of the public ComputerCraft API - http://www.computercraft.info | ||||||
|  * Copyright Daniel Ratcliffe, 2011-2019. Do not distribute without permission. |  * Copyright Daniel Ratcliffe, 2011-2019. This API may be redistributed unmodified and in full only. | ||||||
|  * Send enquiries to dratcliffe@gmail.com |  * For help using the API, and posting your mods, visit the forums at computercraft.info. | ||||||
|  */ |  */ | ||||||
|  |  | ||||||
| package dan200.computercraft.api.peripheral; | package dan200.computercraft.api.peripheral; | ||||||
|  |  | ||||||
| import javax.annotation.Nonnull; | import javax.annotation.Nonnull; | ||||||
|   | |||||||
| @@ -1,9 +1,8 @@ | |||||||
| /* | /* | ||||||
|  * This file is part of ComputerCraft - http://www.computercraft.info |  * This file is part of the public ComputerCraft API - http://www.computercraft.info | ||||||
|  * Copyright Daniel Ratcliffe, 2011-2019. Do not distribute without permission. |  * Copyright Daniel Ratcliffe, 2011-2019. This API may be redistributed unmodified and in full only. | ||||||
|  * Send enquiries to dratcliffe@gmail.com |  * For help using the API, and posting your mods, visit the forums at computercraft.info. | ||||||
|  */ |  */ | ||||||
|  |  | ||||||
| package dan200.computercraft.api.pocket; | package dan200.computercraft.api.pocket; | ||||||
|  |  | ||||||
| import net.minecraft.item.ItemStack; | import net.minecraft.item.ItemStack; | ||||||
|   | |||||||
| @@ -8,7 +8,7 @@ package dan200.computercraft.api.pocket; | |||||||
|  |  | ||||||
| import dan200.computercraft.api.peripheral.IPeripheral; | import dan200.computercraft.api.peripheral.IPeripheral; | ||||||
| import net.minecraft.entity.Entity; | import net.minecraft.entity.Entity; | ||||||
| import net.minecraft.nbt.NBTTagCompound; | import net.minecraft.nbt.CompoundNBT; | ||||||
| import net.minecraft.util.ResourceLocation; | import net.minecraft.util.ResourceLocation; | ||||||
|  |  | ||||||
| import javax.annotation.Nonnull; | import javax.annotation.Nonnull; | ||||||
| @@ -16,7 +16,7 @@ import javax.annotation.Nullable; | |||||||
| import java.util.Map; | import java.util.Map; | ||||||
|  |  | ||||||
| /** | /** | ||||||
|  * Wrapper class for pocket computers |  * Wrapper class for pocket computers. | ||||||
|  */ |  */ | ||||||
| public interface IPocketAccess | public interface IPocketAccess | ||||||
| { | { | ||||||
| @@ -75,7 +75,7 @@ public interface IPocketAccess | |||||||
|      * @see #updateUpgradeNBTData() |      * @see #updateUpgradeNBTData() | ||||||
|      */ |      */ | ||||||
|     @Nonnull |     @Nonnull | ||||||
|     NBTTagCompound getUpgradeNBTData(); |     CompoundNBT getUpgradeNBTData(); | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * Mark the upgrade-specific NBT as dirty. |      * Mark the upgrade-specific NBT as dirty. | ||||||
|   | |||||||
| @@ -6,7 +6,7 @@ | |||||||
|  |  | ||||||
| package dan200.computercraft.api.redstone; | package dan200.computercraft.api.redstone; | ||||||
|  |  | ||||||
| import net.minecraft.util.EnumFacing; | import net.minecraft.util.Direction; | ||||||
| import net.minecraft.util.math.BlockPos; | import net.minecraft.util.math.BlockPos; | ||||||
| import net.minecraft.world.World; | import net.minecraft.world.World; | ||||||
|  |  | ||||||
| @@ -30,5 +30,5 @@ public interface IBundledRedstoneProvider | |||||||
|      * handle this block. |      * handle this block. | ||||||
|      * @see dan200.computercraft.api.ComputerCraftAPI#registerBundledRedstoneProvider(IBundledRedstoneProvider) |      * @see dan200.computercraft.api.ComputerCraftAPI#registerBundledRedstoneProvider(IBundledRedstoneProvider) | ||||||
|      */ |      */ | ||||||
|     int getBundledRedstoneOutput( @Nonnull World world, @Nonnull BlockPos pos, @Nonnull EnumFacing side ); |     int getBundledRedstoneOutput( @Nonnull World world, @Nonnull BlockPos pos, @Nonnull Direction side ); | ||||||
| } | } | ||||||
|   | |||||||
| @@ -11,8 +11,8 @@ import dan200.computercraft.api.lua.ILuaContext; | |||||||
| import dan200.computercraft.api.lua.LuaException; | import dan200.computercraft.api.lua.LuaException; | ||||||
| import dan200.computercraft.api.peripheral.IPeripheral; | import dan200.computercraft.api.peripheral.IPeripheral; | ||||||
| import net.minecraft.inventory.IInventory; | import net.minecraft.inventory.IInventory; | ||||||
| import net.minecraft.nbt.NBTTagCompound; | import net.minecraft.nbt.CompoundNBT; | ||||||
| import net.minecraft.util.EnumFacing; | import net.minecraft.util.Direction; | ||||||
| import net.minecraft.util.math.BlockPos; | import net.minecraft.util.math.BlockPos; | ||||||
| import net.minecraft.util.math.Vec3d; | import net.minecraft.util.math.Vec3d; | ||||||
| import net.minecraft.world.World; | import net.minecraft.world.World; | ||||||
| @@ -83,10 +83,10 @@ public interface ITurtleAccess | |||||||
|      * Returns the world direction the turtle is currently facing. |      * Returns the world direction the turtle is currently facing. | ||||||
|      * |      * | ||||||
|      * @return The world direction the turtle is currently facing. |      * @return The world direction the turtle is currently facing. | ||||||
|      * @see #setDirection(EnumFacing) |      * @see #setDirection(Direction) | ||||||
|      */ |      */ | ||||||
|     @Nonnull |     @Nonnull | ||||||
|     EnumFacing getDirection(); |     Direction getDirection(); | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * Set the direction the turtle is facing. Note that this will not play a rotation animation, you will also need to |      * Set the direction the turtle is facing. Note that this will not play a rotation animation, you will also need to | ||||||
| @@ -95,7 +95,7 @@ public interface ITurtleAccess | |||||||
|      * @param dir The new direction to set. This should be on either the x or z axis (so north, south, east or west). |      * @param dir The new direction to set. This should be on either the x or z axis (so north, south, east or west). | ||||||
|      * @see #getDirection() |      * @see #getDirection() | ||||||
|      */ |      */ | ||||||
|     void setDirection( @Nonnull EnumFacing dir ); |     void setDirection( @Nonnull Direction dir ); | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * Get the currently selected slot in the turtle's inventory. |      * Get the currently selected slot in the turtle's inventory. | ||||||
| @@ -144,7 +144,9 @@ public interface ITurtleAccess | |||||||
|     GameProfile getOwningPlayer(); |     GameProfile getOwningPlayer(); | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * Get the inventory of this turtle |      * Get the inventory of this turtle. | ||||||
|  |      * | ||||||
|  |      * Note: this inventory should only be accessed and modified on the server thread. | ||||||
|      * |      * | ||||||
|      * @return This turtle's inventory |      * @return This turtle's inventory | ||||||
|      * @see #getItemHandler() |      * @see #getItemHandler() | ||||||
| @@ -155,6 +157,8 @@ public interface ITurtleAccess | |||||||
|     /** |     /** | ||||||
|      * Get the inventory of this turtle as an {@link IItemHandlerModifiable}. |      * Get the inventory of this turtle as an {@link IItemHandlerModifiable}. | ||||||
|      * |      * | ||||||
|  |      * Note: this inventory should only be accessed and modified on the server thread. | ||||||
|  |      * | ||||||
|      * @return This turtle's inventory |      * @return This turtle's inventory | ||||||
|      * @see #getInventory() |      * @see #getInventory() | ||||||
|      * @see IItemHandlerModifiable |      * @see IItemHandlerModifiable | ||||||
| @@ -290,7 +294,7 @@ public interface ITurtleAccess | |||||||
|      * @see #updateUpgradeNBTData(TurtleSide) |      * @see #updateUpgradeNBTData(TurtleSide) | ||||||
|      */ |      */ | ||||||
|     @Nonnull |     @Nonnull | ||||||
|     NBTTagCompound getUpgradeNBTData( @Nullable TurtleSide side ); |     CompoundNBT getUpgradeNBTData( @Nullable TurtleSide side ); | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * Mark the upgrade-specific data as dirty on a specific side. This is required for the data to be synced to the |      * Mark the upgrade-specific data as dirty on a specific side. This is required for the data to be synced to the | ||||||
|   | |||||||
| @@ -13,7 +13,7 @@ import dan200.computercraft.api.turtle.event.TurtleBlockEvent; | |||||||
| import net.minecraft.client.renderer.model.IBakedModel; | import net.minecraft.client.renderer.model.IBakedModel; | ||||||
| import net.minecraft.client.renderer.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.Direction; | ||||||
| import net.minecraft.util.ResourceLocation; | import net.minecraft.util.ResourceLocation; | ||||||
| import net.minecraftforge.api.distmarker.Dist; | import net.minecraftforge.api.distmarker.Dist; | ||||||
| import net.minecraftforge.api.distmarker.OnlyIn; | import net.minecraftforge.api.distmarker.OnlyIn; | ||||||
| @@ -98,8 +98,8 @@ public interface ITurtleUpgrade | |||||||
|      * Will only be called for Tool turtle. Called when turtle.dig() or turtle.attack() is called |      * Will only be called for Tool turtle. Called when turtle.dig() or turtle.attack() is called | ||||||
|      * by the turtle, and the tool is required to do some work. |      * by the turtle, and the tool is required to do some work. | ||||||
|      * |      * | ||||||
|      * Conforming implementations should fire {@link BlockEvent.BreakEvent} and {@link TurtleBlockEvent.Dig}for digging, |      * Conforming implementations should fire {@link BlockEvent.BreakEvent} and {@link TurtleBlockEvent.Dig} for | ||||||
|      * {@link AttackEntityEvent} and {@link TurtleAttackEvent} for attacking. |      * digging, {@link AttackEntityEvent} and {@link TurtleAttackEvent} for attacking. | ||||||
|      * |      * | ||||||
|      * @param turtle    Access to the turtle that the tool resides on. |      * @param turtle    Access to the turtle that the tool resides on. | ||||||
|      * @param side      Which side of the turtle (left or right) the tool resides on. |      * @param side      Which side of the turtle (left or right) the tool resides on. | ||||||
| @@ -113,7 +113,7 @@ public interface ITurtleUpgrade | |||||||
|      * to be called. |      * to be called. | ||||||
|      */ |      */ | ||||||
|     @Nonnull |     @Nonnull | ||||||
|     default TurtleCommandResult useTool( @Nonnull ITurtleAccess turtle, @Nonnull TurtleSide side, @Nonnull TurtleVerb verb, @Nonnull EnumFacing direction ) |     default TurtleCommandResult useTool( @Nonnull ITurtleAccess turtle, @Nonnull TurtleSide side, @Nonnull TurtleVerb verb, @Nonnull Direction direction ) | ||||||
|     { |     { | ||||||
|         return TurtleCommandResult.failure(); |         return TurtleCommandResult.failure(); | ||||||
|     } |     } | ||||||
|   | |||||||
| @@ -6,7 +6,7 @@ | |||||||
|  |  | ||||||
| package dan200.computercraft.api.turtle; | package dan200.computercraft.api.turtle; | ||||||
|  |  | ||||||
| import net.minecraft.util.EnumFacing; | import net.minecraft.util.Direction; | ||||||
|  |  | ||||||
| import javax.annotation.Nonnull; | import javax.annotation.Nonnull; | ||||||
| import javax.annotation.Nullable; | import javax.annotation.Nullable; | ||||||
| @@ -15,7 +15,7 @@ import javax.annotation.Nullable; | |||||||
|  * Used to indicate the result of executing a turtle command. |  * Used to indicate the result of executing a turtle command. | ||||||
|  * |  * | ||||||
|  * @see ITurtleCommand#execute(ITurtleAccess) |  * @see ITurtleCommand#execute(ITurtleAccess) | ||||||
|  * @see ITurtleUpgrade#useTool(ITurtleAccess, TurtleSide, TurtleVerb, EnumFacing) |  * @see ITurtleUpgrade#useTool(ITurtleAccess, TurtleSide, TurtleVerb, Direction) | ||||||
|  */ |  */ | ||||||
| public final class TurtleCommandResult | public final class TurtleCommandResult | ||||||
| { | { | ||||||
|   | |||||||
| @@ -12,12 +12,12 @@ package dan200.computercraft.api.turtle; | |||||||
| public enum TurtleSide | public enum TurtleSide | ||||||
| { | { | ||||||
|     /** |     /** | ||||||
|      * The turtle's left side (where the pickaxe usually is on a Wireless Mining Turtle) |      * The turtle's left side (where the pickaxe usually is on a Wireless Mining Turtle). | ||||||
|      */ |      */ | ||||||
|     Left, |     Left, | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * The turtle's right side (where the modem usually is on a Wireless Mining Turtle) |      * The turtle's right side (where the modem usually is on a Wireless Mining Turtle). | ||||||
|      */ |      */ | ||||||
|     Right, |     Right, | ||||||
| } | } | ||||||
|   | |||||||
| @@ -6,24 +6,24 @@ | |||||||
|  |  | ||||||
| package dan200.computercraft.api.turtle; | package dan200.computercraft.api.turtle; | ||||||
|  |  | ||||||
| import net.minecraft.util.EnumFacing; | import net.minecraft.util.Direction; | ||||||
|  |  | ||||||
| /** | /** | ||||||
|  * An enum representing the different actions that an {@link ITurtleUpgrade} of type Tool may be called on to perform by |  * An enum representing the different actions that an {@link ITurtleUpgrade} of type Tool may be called on to perform by | ||||||
|  * a turtle. |  * a turtle. | ||||||
|  * |  * | ||||||
|  * @see ITurtleUpgrade#getType() |  * @see ITurtleUpgrade#getType() | ||||||
|  * @see ITurtleUpgrade#useTool(ITurtleAccess, TurtleSide, TurtleVerb, EnumFacing) |  * @see ITurtleUpgrade#useTool(ITurtleAccess, TurtleSide, TurtleVerb, Direction) | ||||||
|  */ |  */ | ||||||
| public enum TurtleVerb | public enum TurtleVerb | ||||||
| { | { | ||||||
|     /** |     /** | ||||||
|      * The turtle called {@code turtle.dig()}, {@code turtle.digUp()} or {@code turtle.digDown()} |      * The turtle called {@code turtle.dig()}, {@code turtle.digUp()} or {@code turtle.digDown()}. | ||||||
|      */ |      */ | ||||||
|     Dig, |     Dig, | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * The turtle called {@code turtle.attack()}, {@code turtle.attackUp()} or {@code turtle.attackDown()} |      * The turtle called {@code turtle.attack()}, {@code turtle.attackUp()} or {@code turtle.attackDown()}. | ||||||
|      */ |      */ | ||||||
|     Attack, |     Attack, | ||||||
| } | } | ||||||
|   | |||||||
| @@ -71,7 +71,7 @@ public enum TurtleAction | |||||||
|     EQUIP, |     EQUIP, | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * Inspect a block in world |      * Inspect a block in world. | ||||||
|      * |      * | ||||||
|      * @see TurtleBlockEvent.Inspect |      * @see TurtleBlockEvent.Inspect | ||||||
|      */ |      */ | ||||||
|   | |||||||
| @@ -11,7 +11,7 @@ import dan200.computercraft.api.turtle.ITurtleUpgrade; | |||||||
| import dan200.computercraft.api.turtle.TurtleSide; | import dan200.computercraft.api.turtle.TurtleSide; | ||||||
| import dan200.computercraft.api.turtle.TurtleVerb; | import dan200.computercraft.api.turtle.TurtleVerb; | ||||||
| import net.minecraft.entity.Entity; | import net.minecraft.entity.Entity; | ||||||
| import net.minecraft.util.EnumFacing; | import net.minecraft.util.Direction; | ||||||
| import net.minecraftforge.common.util.FakePlayer; | import net.minecraftforge.common.util.FakePlayer; | ||||||
| import net.minecraftforge.event.entity.player.AttackEntityEvent; | import net.minecraftforge.event.entity.player.AttackEntityEvent; | ||||||
|  |  | ||||||
| @@ -21,7 +21,7 @@ import java.util.Objects; | |||||||
| /** | /** | ||||||
|  * Fired when a turtle attempts to attack an entity. |  * Fired when a turtle attempts to attack an entity. | ||||||
|  * |  * | ||||||
|  * This must be fired by {@link ITurtleUpgrade#useTool(ITurtleAccess, TurtleSide, TurtleVerb, EnumFacing)}, |  * This must be fired by {@link ITurtleUpgrade#useTool(ITurtleAccess, TurtleSide, TurtleVerb, Direction)}, | ||||||
|  * as the base {@code turtle.attack()} command does not fire it. |  * as the base {@code turtle.attack()} command does not fire it. | ||||||
|  * |  * | ||||||
|  * Note that such commands should also fire {@link AttackEntityEvent}, so you do not need to listen to both. |  * Note that such commands should also fire {@link AttackEntityEvent}, so you do not need to listen to both. | ||||||
|   | |||||||
| @@ -12,9 +12,9 @@ import dan200.computercraft.api.turtle.ITurtleAccess; | |||||||
| 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.api.turtle.TurtleVerb; | import dan200.computercraft.api.turtle.TurtleVerb; | ||||||
| import net.minecraft.block.state.IBlockState; | import net.minecraft.block.BlockState; | ||||||
| import net.minecraft.item.ItemStack; | import net.minecraft.item.ItemStack; | ||||||
| import net.minecraft.util.EnumFacing; | import net.minecraft.util.Direction; | ||||||
| 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.common.util.FakePlayer; | import net.minecraftforge.common.util.FakePlayer; | ||||||
| @@ -75,7 +75,7 @@ public abstract class TurtleBlockEvent extends TurtlePlayerEvent | |||||||
|     /** |     /** | ||||||
|      * Fired when a turtle attempts to dig a block. |      * Fired when a turtle attempts to dig a block. | ||||||
|      * |      * | ||||||
|      * This must be fired by {@link ITurtleUpgrade#useTool(ITurtleAccess, TurtleSide, TurtleVerb, EnumFacing)}, |      * This must be fired by {@link ITurtleUpgrade#useTool(ITurtleAccess, TurtleSide, TurtleVerb, Direction)}, | ||||||
|      * as the base {@code turtle.dig()} command does not fire it. |      * as the base {@code turtle.dig()} command does not fire it. | ||||||
|      * |      * | ||||||
|      * Note that such commands should also fire {@link BlockEvent.BreakEvent}, so you do not need to listen to both. |      * Note that such commands should also fire {@link BlockEvent.BreakEvent}, so you do not need to listen to both. | ||||||
| @@ -84,11 +84,11 @@ public abstract class TurtleBlockEvent extends TurtlePlayerEvent | |||||||
|      */ |      */ | ||||||
|     public static class Dig extends TurtleBlockEvent |     public static class Dig extends TurtleBlockEvent | ||||||
|     { |     { | ||||||
|         private final IBlockState block; |         private final BlockState block; | ||||||
|         private final ITurtleUpgrade upgrade; |         private final ITurtleUpgrade upgrade; | ||||||
|         private final TurtleSide side; |         private final TurtleSide side; | ||||||
|  |  | ||||||
|         public Dig( @Nonnull ITurtleAccess turtle, @Nonnull FakePlayer player, @Nonnull World world, @Nonnull BlockPos pos, @Nonnull IBlockState block, @Nonnull ITurtleUpgrade upgrade, @Nonnull TurtleSide side ) |         public Dig( @Nonnull ITurtleAccess turtle, @Nonnull FakePlayer player, @Nonnull World world, @Nonnull BlockPos pos, @Nonnull BlockState block, @Nonnull ITurtleUpgrade upgrade, @Nonnull TurtleSide side ) | ||||||
|         { |         { | ||||||
|             super( turtle, TurtleAction.DIG, player, world, pos ); |             super( turtle, TurtleAction.DIG, player, world, pos ); | ||||||
|  |  | ||||||
| @@ -106,13 +106,13 @@ public abstract class TurtleBlockEvent extends TurtlePlayerEvent | |||||||
|          * @return The block which is going to be broken. |          * @return The block which is going to be broken. | ||||||
|          */ |          */ | ||||||
|         @Nonnull |         @Nonnull | ||||||
|         public IBlockState getBlock() |         public BlockState getBlock() | ||||||
|         { |         { | ||||||
|             return block; |             return block; | ||||||
|         } |         } | ||||||
|  |  | ||||||
|         /** |         /** | ||||||
|          * Get the upgrade doing the digging |          * Get the upgrade doing the digging. | ||||||
|          * |          * | ||||||
|          * @return The upgrade doing the digging. |          * @return The upgrade doing the digging. | ||||||
|          */ |          */ | ||||||
| @@ -185,10 +185,10 @@ public abstract class TurtleBlockEvent extends TurtlePlayerEvent | |||||||
|      */ |      */ | ||||||
|     public static class Inspect extends TurtleBlockEvent |     public static class Inspect extends TurtleBlockEvent | ||||||
|     { |     { | ||||||
|         private final IBlockState state; |         private final BlockState state; | ||||||
|         private final Map<String, Object> data; |         private final Map<String, Object> data; | ||||||
|  |  | ||||||
|         public Inspect( @Nonnull ITurtleAccess turtle, @Nonnull FakePlayer player, @Nonnull World world, @Nonnull BlockPos pos, @Nonnull IBlockState state, @Nonnull Map<String, Object> data ) |         public Inspect( @Nonnull ITurtleAccess turtle, @Nonnull FakePlayer player, @Nonnull World world, @Nonnull BlockPos pos, @Nonnull BlockState state, @Nonnull Map<String, Object> data ) | ||||||
|         { |         { | ||||||
|             super( turtle, TurtleAction.INSPECT, player, world, pos ); |             super( turtle, TurtleAction.INSPECT, player, world, pos ); | ||||||
|  |  | ||||||
| @@ -204,7 +204,7 @@ public abstract class TurtleBlockEvent extends TurtlePlayerEvent | |||||||
|          * @return The inspected block state. |          * @return The inspected block state. | ||||||
|          */ |          */ | ||||||
|         @Nonnull |         @Nonnull | ||||||
|         public IBlockState getState() |         public BlockState getState() | ||||||
|         { |         { | ||||||
|             return state; |             return state; | ||||||
|         } |         } | ||||||
|   | |||||||
| @@ -1,9 +1,8 @@ | |||||||
| /* | /* | ||||||
|  * This file is part of ComputerCraft - http://www.computercraft.info |  * This file is part of the public ComputerCraft API - http://www.computercraft.info | ||||||
|  * Copyright Daniel Ratcliffe, 2011-2019. Do not distribute without permission. |  * Copyright Daniel Ratcliffe, 2011-2019. This API may be redistributed unmodified and in full only. | ||||||
|  * Send enquiries to dratcliffe@gmail.com |  * For help using the API, and posting your mods, visit the forums at computercraft.info. | ||||||
|  */ |  */ | ||||||
|  |  | ||||||
| package dan200.computercraft.api.turtle.event; | package dan200.computercraft.api.turtle.event; | ||||||
|  |  | ||||||
| import dan200.computercraft.api.lua.ILuaContext; | import dan200.computercraft.api.lua.ILuaContext; | ||||||
|   | |||||||
| @@ -31,7 +31,7 @@ public abstract class TurtleInventoryEvent extends TurtleBlockEvent | |||||||
|     } |     } | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * Get the inventory being interacted with |      * Get the inventory being interacted with. | ||||||
|      * |      * | ||||||
|      * @return The inventory being interacted with, {@code null} if the item will be dropped to/sucked from the world. |      * @return The inventory being interacted with, {@code null} if the item will be dropped to/sucked from the world. | ||||||
|      */ |      */ | ||||||
|   | |||||||
| @@ -1,9 +1,8 @@ | |||||||
| /* | /* | ||||||
|  * This file is part of ComputerCraft - http://www.computercraft.info |  * This file is part of the public ComputerCraft API - http://www.computercraft.info | ||||||
|  * Copyright Daniel Ratcliffe, 2011-2019. Do not distribute without permission. |  * Copyright Daniel Ratcliffe, 2011-2019. This API may be redistributed unmodified and in full only. | ||||||
|  * Send enquiries to dratcliffe@gmail.com |  * For help using the API, and posting your mods, visit the forums at computercraft.info. | ||||||
|  */ |  */ | ||||||
|  |  | ||||||
| package dan200.computercraft.api.turtle.event; | package dan200.computercraft.api.turtle.event; | ||||||
|  |  | ||||||
| import dan200.computercraft.api.turtle.ITurtleAccess; | import dan200.computercraft.api.turtle.ITurtleAccess; | ||||||
|   | |||||||
| @@ -16,15 +16,14 @@ import net.minecraft.client.Minecraft; | |||||||
| import net.minecraft.client.renderer.model.IBakedModel; | import net.minecraft.client.renderer.model.IBakedModel; | ||||||
| import net.minecraft.client.renderer.model.IUnbakedModel; | import net.minecraft.client.renderer.model.IUnbakedModel; | ||||||
| import net.minecraft.client.renderer.model.ModelResourceLocation; | import net.minecraft.client.renderer.model.ModelResourceLocation; | ||||||
| import net.minecraft.client.renderer.model.ModelRotation; |  | ||||||
| import net.minecraft.client.renderer.vertex.DefaultVertexFormats; | import net.minecraft.client.renderer.vertex.DefaultVertexFormats; | ||||||
| import net.minecraft.resources.IResourceManager; |  | ||||||
| import net.minecraft.util.ResourceLocation; | import net.minecraft.util.ResourceLocation; | ||||||
| import net.minecraftforge.api.distmarker.Dist; | 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.BasicState; | ||||||
| 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.eventbus.api.SubscribeEvent; | ||||||
| @@ -36,7 +35,7 @@ import java.util.Map; | |||||||
| /** | /** | ||||||
|  * Registers textures and models for items. |  * Registers textures and models for items. | ||||||
|  */ |  */ | ||||||
| @Mod.EventBusSubscriber( modid = ComputerCraft.MOD_ID, value = Dist.CLIENT ) | @Mod.EventBusSubscriber( modid = ComputerCraft.MOD_ID, value = Dist.CLIENT, bus = Mod.EventBusSubscriber.Bus.MOD ) | ||||||
| public final class ClientRegistry | public final class ClientRegistry | ||||||
| { | { | ||||||
|     private static final String[] EXTRA_MODELS = new String[] { |     private static final String[] EXTRA_MODELS = new String[] { | ||||||
| @@ -60,8 +59,8 @@ public final class ClientRegistry | |||||||
|     }; |     }; | ||||||
|  |  | ||||||
|     private static final String[] EXTRA_TEXTURES = new String[] { |     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 |         // TODO: Gather these automatically from the model. Sadly the model loader isn't available | ||||||
|         //  model loading code. |         //  when stitching textures. | ||||||
|         "block/turtle_colour", |         "block/turtle_colour", | ||||||
|         "block/turtle_elf_overlay", |         "block/turtle_elf_overlay", | ||||||
|         "block/turtle_crafty_face", |         "block/turtle_crafty_face", | ||||||
| @@ -79,10 +78,11 @@ public final class ClientRegistry | |||||||
|     @SubscribeEvent |     @SubscribeEvent | ||||||
|     public static void onTextureStitchEvent( TextureStitchEvent.Pre event ) |     public static void onTextureStitchEvent( TextureStitchEvent.Pre event ) | ||||||
|     { |     { | ||||||
|         IResourceManager manager = Minecraft.getInstance().getResourceManager(); |         if( event.getMap() != Minecraft.getInstance().getTextureMap() ) return; | ||||||
|  |  | ||||||
|         for( String extra : EXTRA_TEXTURES ) |         for( String extra : EXTRA_TEXTURES ) | ||||||
|         { |         { | ||||||
|             event.getMap().registerSprite( manager, new ResourceLocation( ComputerCraft.MOD_ID, extra ) ); |             event.addSprite( new ResourceLocation( ComputerCraft.MOD_ID, extra ) ); | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  |  | ||||||
| @@ -91,7 +91,7 @@ public final class ClientRegistry | |||||||
|     { |     { | ||||||
|         // Load all extra models |         // Load all extra models | ||||||
|         ModelLoader loader = event.getModelLoader(); |         ModelLoader loader = event.getModelLoader(); | ||||||
|         Map<ModelResourceLocation, IBakedModel> registry = event.getModelRegistry(); |         Map<ResourceLocation, IBakedModel> registry = event.getModelRegistry(); | ||||||
|  |  | ||||||
|         for( String model : EXTRA_MODELS ) |         for( String model : EXTRA_MODELS ) | ||||||
|         { |         { | ||||||
| @@ -160,9 +160,9 @@ public final class ClientRegistry | |||||||
|         model.getTextures( loader::getUnbakedModel, new HashSet<>() ); |         model.getTextures( loader::getUnbakedModel, new HashSet<>() ); | ||||||
|  |  | ||||||
|         return model.bake( |         return model.bake( | ||||||
|             loader::getUnbakedModel, |             loader, | ||||||
|             ModelLoader.defaultTextureGetter(), |             ModelLoader.defaultTextureGetter(), | ||||||
|             ModelRotation.X0_Y0, false, DefaultVertexFormats.BLOCK |             new BasicState( model.getDefaultState(), false ), DefaultVertexFormats.BLOCK | ||||||
|         ); |         ); | ||||||
|     } |     } | ||||||
| } | } | ||||||
|   | |||||||
| @@ -12,8 +12,8 @@ import dan200.computercraft.shared.command.text.TableFormatter; | |||||||
| import it.unimi.dsi.fastutil.ints.Int2IntOpenHashMap; | import it.unimi.dsi.fastutil.ints.Int2IntOpenHashMap; | ||||||
| import net.minecraft.client.Minecraft; | import net.minecraft.client.Minecraft; | ||||||
| import net.minecraft.client.gui.FontRenderer; | import net.minecraft.client.gui.FontRenderer; | ||||||
| import net.minecraft.client.gui.GuiNewChat; | import net.minecraft.client.gui.NewChatGui; | ||||||
| import net.minecraft.client.gui.GuiUtilRenderComponents; | import net.minecraft.client.gui.RenderComponentsUtil; | ||||||
| import net.minecraft.util.math.MathHelper; | import net.minecraft.util.math.MathHelper; | ||||||
| import net.minecraft.util.text.ITextComponent; | import net.minecraft.util.text.ITextComponent; | ||||||
| import net.minecraft.util.text.TextFormatting; | import net.minecraft.util.text.TextFormatting; | ||||||
| @@ -65,18 +65,18 @@ public class ClientTableFormatter implements TableFormatter | |||||||
|     public void writeLine( int id, ITextComponent component ) |     public void writeLine( int id, ITextComponent component ) | ||||||
|     { |     { | ||||||
|         Minecraft mc = Minecraft.getInstance(); |         Minecraft mc = Minecraft.getInstance(); | ||||||
|         GuiNewChat chat = mc.ingameGUI.getChatGUI(); |         NewChatGui 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.getScale() ); |         int maxWidth = MathHelper.floor( chat.getChatWidth() / chat.getScale() ); | ||||||
|         List<ITextComponent> list = GuiUtilRenderComponents.splitText( component, maxWidth, mc.fontRenderer, false, false ); |         List<ITextComponent> list = RenderComponentsUtil.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 ); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @Override |     @Override | ||||||
|     public int display( TableBuilder table ) |     public int display( TableBuilder table ) | ||||||
|     { |     { | ||||||
|         GuiNewChat chat = Minecraft.getInstance().ingameGUI.getChatGUI(); |         NewChatGui chat = Minecraft.getInstance().ingameGUI.getChatGUI(); | ||||||
|  |  | ||||||
|         int lastHeight = lastHeights.get( table.getId() ); |         int lastHeight = lastHeights.get( table.getId() ); | ||||||
|  |  | ||||||
|   | |||||||
| @@ -8,9 +8,9 @@ package dan200.computercraft.client; | |||||||
|  |  | ||||||
| import dan200.computercraft.ComputerCraft; | import dan200.computercraft.ComputerCraft; | ||||||
| import net.minecraftforge.api.distmarker.Dist; | import net.minecraftforge.api.distmarker.Dist; | ||||||
|  | import net.minecraftforge.event.TickEvent; | ||||||
| import net.minecraftforge.eventbus.api.SubscribeEvent; | import net.minecraftforge.eventbus.api.SubscribeEvent; | ||||||
| import net.minecraftforge.fml.common.Mod; | import net.minecraftforge.fml.common.Mod; | ||||||
| import net.minecraftforge.fml.common.gameevent.TickEvent; |  | ||||||
|  |  | ||||||
| @Mod.EventBusSubscriber( modid = ComputerCraft.MOD_ID, value = Dist.CLIENT ) | @Mod.EventBusSubscriber( modid = ComputerCraft.MOD_ID, value = Dist.CLIENT ) | ||||||
| public final class FrameInfo | public final class FrameInfo | ||||||
|   | |||||||
| @@ -6,11 +6,11 @@ | |||||||
|  |  | ||||||
| package dan200.computercraft.client.gui; | package dan200.computercraft.client.gui; | ||||||
|  |  | ||||||
|  | import com.mojang.blaze3d.platform.GlStateManager; | ||||||
| import dan200.computercraft.core.terminal.TextBuffer; | import dan200.computercraft.core.terminal.TextBuffer; | ||||||
| import dan200.computercraft.shared.util.Palette; | import dan200.computercraft.shared.util.Palette; | ||||||
| import net.minecraft.client.Minecraft; | import net.minecraft.client.Minecraft; | ||||||
| import net.minecraft.client.renderer.BufferBuilder; | import net.minecraft.client.renderer.BufferBuilder; | ||||||
| import net.minecraft.client.renderer.GlStateManager; |  | ||||||
| import net.minecraft.client.renderer.Tessellator; | import net.minecraft.client.renderer.Tessellator; | ||||||
| import net.minecraft.client.renderer.texture.TextureManager; | import net.minecraft.client.renderer.texture.TextureManager; | ||||||
| import net.minecraft.client.renderer.vertex.DefaultVertexFormats; | import net.minecraft.client.renderer.vertex.DefaultVertexFormats; | ||||||
| @@ -129,9 +129,9 @@ public final class FixedWidthFontRenderer | |||||||
|             } |             } | ||||||
|             drawQuad( renderer, x + i * FONT_WIDTH, y, colour, FONT_WIDTH, p, greyScale ); |             drawQuad( renderer, x + i * FONT_WIDTH, y, colour, FONT_WIDTH, p, greyScale ); | ||||||
|         } |         } | ||||||
|         GlStateManager.disableTexture2D(); |         GlStateManager.disableTexture(); | ||||||
|         tessellator.draw(); |         tessellator.draw(); | ||||||
|         GlStateManager.enableTexture2D(); |         GlStateManager.enableTexture(); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     public void drawStringTextPart( int x, int y, TextBuffer s, TextBuffer textColour, boolean greyScale, Palette p ) |     public void drawStringTextPart( int x, int y, TextBuffer s, TextBuffer textColour, boolean greyScale, Palette p ) | ||||||
| @@ -195,6 +195,6 @@ public final class FixedWidthFontRenderer | |||||||
|     public void bindFont() |     public void bindFont() | ||||||
|     { |     { | ||||||
|         m_textureManager.bindTexture( FONT ); |         m_textureManager.bindTexture( FONT ); | ||||||
|         GlStateManager.texParameteri( GL11.GL_TEXTURE_2D, GL11.GL_TEXTURE_WRAP_S, GL11.GL_CLAMP ); |         GlStateManager.texParameter( GL11.GL_TEXTURE_2D, GL11.GL_TEXTURE_WRAP_S, GL11.GL_CLAMP ); | ||||||
|     } |     } | ||||||
| } | } | ||||||
|   | |||||||
| @@ -6,19 +6,23 @@ | |||||||
|  |  | ||||||
| package dan200.computercraft.client.gui; | package dan200.computercraft.client.gui; | ||||||
|  |  | ||||||
|  | import com.mojang.blaze3d.platform.GlStateManager; | ||||||
| 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.client.gui.widgets.WidgetWrapper; | ||||||
| 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.inventory.ContainerComputer; | import dan200.computercraft.shared.computer.inventory.ContainerComputer; | ||||||
| import net.minecraft.client.gui.inventory.GuiContainer; | import dan200.computercraft.shared.computer.inventory.ContainerComputerBase; | ||||||
| import net.minecraft.client.renderer.GlStateManager; | import dan200.computercraft.shared.computer.inventory.ContainerViewComputer; | ||||||
| import net.minecraft.inventory.Container; | import dan200.computercraft.shared.pocket.inventory.ContainerPocketComputer; | ||||||
|  | import net.minecraft.client.gui.screen.inventory.ContainerScreen; | ||||||
|  | import net.minecraft.entity.player.PlayerInventory; | ||||||
| import net.minecraft.util.ResourceLocation; | import net.minecraft.util.ResourceLocation; | ||||||
|  | import net.minecraft.util.text.ITextComponent; | ||||||
|  | import org.lwjgl.glfw.GLFW; | ||||||
|  |  | ||||||
| public class GuiComputer extends GuiContainer | public final class GuiComputer<T extends ContainerComputerBase> extends ContainerScreen<T> | ||||||
| { | { | ||||||
|     public static final ResourceLocation BACKGROUND_NORMAL = new ResourceLocation( ComputerCraft.MOD_ID, "textures/gui/corners_normal.png" ); |     public static final ResourceLocation BACKGROUND_NORMAL = new ResourceLocation( ComputerCraft.MOD_ID, "textures/gui/corners_normal.png" ); | ||||||
|     public static final ResourceLocation BACKGROUND_ADVANCED = new ResourceLocation( ComputerCraft.MOD_ID, "textures/gui/corners_advanced.png" ); |     public static final ResourceLocation BACKGROUND_ADVANCED = new ResourceLocation( ComputerCraft.MOD_ID, "textures/gui/corners_advanced.png" ); | ||||||
| @@ -33,31 +37,47 @@ public class GuiComputer extends GuiContainer | |||||||
|     private WidgetTerminal terminal; |     private WidgetTerminal terminal; | ||||||
|     private WidgetWrapper terminalWrapper; |     private WidgetWrapper terminalWrapper; | ||||||
|  |  | ||||||
|     public GuiComputer( Container container, ComputerFamily family, ClientComputer computer, int termWidth, int termHeight ) |     private GuiComputer( | ||||||
|  |         T container, PlayerInventory player, ITextComponent title, int termWidth, int termHeight | ||||||
|  |     ) | ||||||
|     { |     { | ||||||
|         super( container ); |         super( container, player, title ); | ||||||
|         m_family = family; |         m_family = container.getFamily(); | ||||||
|         m_computer = computer; |         m_computer = (ClientComputer) container.getComputer(); | ||||||
|         m_termWidth = termWidth; |         m_termWidth = termWidth; | ||||||
|         m_termHeight = termHeight; |         m_termHeight = termHeight; | ||||||
|         terminal = null; |         terminal = null; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     public GuiComputer( TileComputer computer ) |     public static GuiComputer<ContainerComputer> create( ContainerComputer container, PlayerInventory inventory, ITextComponent component ) | ||||||
|     { |     { | ||||||
|         this( |         return new GuiComputer<>( | ||||||
|             new ContainerComputer( computer ), |             container, inventory, component, | ||||||
|             computer.getFamily(), |             ComputerCraft.terminalWidth_computer, ComputerCraft.terminalHeight_computer | ||||||
|             computer.createClientComputer(), |  | ||||||
|             ComputerCraft.terminalWidth_computer, |  | ||||||
|             ComputerCraft.terminalHeight_computer |  | ||||||
|         ); |         ); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @Override |     public static GuiComputer<ContainerPocketComputer> createPocket( ContainerPocketComputer container, PlayerInventory inventory, ITextComponent component ) | ||||||
|     protected void initGui() |  | ||||||
|     { |     { | ||||||
|         mc.keyboardListener.enableRepeatEvents( true ); |         return new GuiComputer<>( | ||||||
|  |             container, inventory, component, | ||||||
|  |             ComputerCraft.terminalWidth_pocketComputer, ComputerCraft.terminalHeight_pocketComputer | ||||||
|  |         ); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     public static GuiComputer<ContainerViewComputer> createView( ContainerViewComputer container, PlayerInventory inventory, ITextComponent component ) | ||||||
|  |     { | ||||||
|  |         return new GuiComputer<>( | ||||||
|  |             container, inventory, component, | ||||||
|  |             container.getWidth(), container.getHeight() | ||||||
|  |         ); | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |  | ||||||
|  |     @Override | ||||||
|  |     protected void init() | ||||||
|  |     { | ||||||
|  |         minecraft.keyboardListener.enableRepeatEvents( true ); | ||||||
|  |  | ||||||
|         int termPxWidth = m_termWidth * FixedWidthFontRenderer.FONT_WIDTH; |         int termPxWidth = m_termWidth * FixedWidthFontRenderer.FONT_WIDTH; | ||||||
|         int termPxHeight = m_termHeight * FixedWidthFontRenderer.FONT_HEIGHT; |         int termPxHeight = m_termHeight * FixedWidthFontRenderer.FONT_HEIGHT; | ||||||
| @@ -65,9 +85,9 @@ public class GuiComputer extends GuiContainer | |||||||
|         xSize = termPxWidth + 4 + 24; |         xSize = termPxWidth + 4 + 24; | ||||||
|         ySize = termPxHeight + 4 + 24; |         ySize = termPxHeight + 4 + 24; | ||||||
|  |  | ||||||
|         super.initGui(); |         super.init(); | ||||||
|  |  | ||||||
|         terminal = new WidgetTerminal( mc, () -> m_computer, m_termWidth, m_termHeight, 2, 2, 2, 2 ); |         terminal = new WidgetTerminal( minecraft, () -> m_computer, m_termWidth, m_termHeight, 2, 2, 2, 2 ); | ||||||
|         terminalWrapper = new WidgetWrapper( terminal, 2 + 12 + guiLeft, 2 + 12 + guiTop, termPxWidth, termPxHeight ); |         terminalWrapper = new WidgetWrapper( terminal, 2 + 12 + guiLeft, 2 + 12 + guiTop, termPxWidth, termPxHeight ); | ||||||
|  |  | ||||||
|         children.add( terminalWrapper ); |         children.add( terminalWrapper ); | ||||||
| @@ -75,12 +95,12 @@ public class GuiComputer extends GuiContainer | |||||||
|     } |     } | ||||||
|  |  | ||||||
|     @Override |     @Override | ||||||
|     public void onGuiClosed() |     public void removed() | ||||||
|     { |     { | ||||||
|         super.onGuiClosed(); |         super.removed(); | ||||||
|         children.remove( terminal ); |         children.remove( terminal ); | ||||||
|         terminal = null; |         terminal = null; | ||||||
|         mc.keyboardListener.enableRepeatEvents( false ); |         minecraft.keyboardListener.enableRepeatEvents( false ); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @Override |     @Override | ||||||
| @@ -90,6 +110,18 @@ public class GuiComputer extends GuiContainer | |||||||
|         terminal.update(); |         terminal.update(); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     @Override | ||||||
|  |     public boolean keyPressed( int key, int scancode, int modifiers ) | ||||||
|  |     { | ||||||
|  |         // Forward the tab key to the terminal, rather than moving between controls. | ||||||
|  |         if( key == GLFW.GLFW_KEY_TAB && getFocused() != null && getFocused() == terminalWrapper ) | ||||||
|  |         { | ||||||
|  |             return getFocused().keyPressed( key, scancode, modifiers ); | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         return super.keyPressed( key, scancode, modifiers ); | ||||||
|  |     } | ||||||
|  |  | ||||||
|     @Override |     @Override | ||||||
|     public void drawGuiContainerBackgroundLayer( float partialTicks, int mouseX, int mouseY ) |     public void drawGuiContainerBackgroundLayer( float partialTicks, int mouseX, int mouseY ) | ||||||
|     { |     { | ||||||
| @@ -108,32 +140,32 @@ public class GuiComputer extends GuiContainer | |||||||
|         { |         { | ||||||
|             case Normal: |             case Normal: | ||||||
|             default: |             default: | ||||||
|                 mc.getTextureManager().bindTexture( BACKGROUND_NORMAL ); |                 minecraft.getTextureManager().bindTexture( BACKGROUND_NORMAL ); | ||||||
|                 break; |                 break; | ||||||
|             case Advanced: |             case Advanced: | ||||||
|                 mc.getTextureManager().bindTexture( BACKGROUND_ADVANCED ); |                 minecraft.getTextureManager().bindTexture( BACKGROUND_ADVANCED ); | ||||||
|                 break; |                 break; | ||||||
|             case Command: |             case Command: | ||||||
|                 mc.getTextureManager().bindTexture( BACKGROUND_COMMAND ); |                 minecraft.getTextureManager().bindTexture( BACKGROUND_COMMAND ); | ||||||
|                 break; |                 break; | ||||||
|         } |         } | ||||||
|  |  | ||||||
|         drawTexturedModalRect( startX - 12, startY - 12, 12, 28, 12, 12 ); |         blit( startX - 12, startY - 12, 12, 28, 12, 12 ); | ||||||
|         drawTexturedModalRect( startX - 12, endY, 12, 40, 12, 12 ); |         blit( startX - 12, endY, 12, 40, 12, 12 ); | ||||||
|         drawTexturedModalRect( endX, startY - 12, 24, 28, 12, 12 ); |         blit( endX, startY - 12, 24, 28, 12, 12 ); | ||||||
|         drawTexturedModalRect( endX, endY, 24, 40, 12, 12 ); |         blit( endX, endY, 24, 40, 12, 12 ); | ||||||
|  |  | ||||||
|         drawTexturedModalRect( startX, startY - 12, 0, 0, endX - startX, 12 ); |         blit( startX, startY - 12, 0, 0, endX - startX, 12 ); | ||||||
|         drawTexturedModalRect( startX, endY, 0, 12, endX - startX, 12 ); |         blit( startX, endY, 0, 12, endX - startX, 12 ); | ||||||
|  |  | ||||||
|         drawTexturedModalRect( startX - 12, startY, 0, 28, 12, endY - startY ); |         blit( startX - 12, startY, 0, 28, 12, endY - startY ); | ||||||
|         drawTexturedModalRect( endX, startY, 36, 28, 12, endY - startY ); |         blit( endX, startY, 36, 28, 12, endY - startY ); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @Override |     @Override | ||||||
|     public void render( int mouseX, int mouseY, float partialTicks ) |     public void render( int mouseX, int mouseY, float partialTicks ) | ||||||
|     { |     { | ||||||
|         drawDefaultBackground(); |         renderBackground(); | ||||||
|         super.render( mouseX, mouseY, partialTicks ); |         super.render( mouseX, mouseY, partialTicks ); | ||||||
|         renderHoveredToolTip( mouseX, mouseY ); |         renderHoveredToolTip( mouseX, mouseY ); | ||||||
|     } |     } | ||||||
| @@ -144,11 +176,4 @@ public class GuiComputer extends GuiContainer | |||||||
|         return (getFocused() != null && getFocused().mouseDragged( x, y, button, deltaX, deltaY )) |         return (getFocused() != null && getFocused().mouseDragged( x, y, button, deltaX, deltaY )) | ||||||
|             || super.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 ); |  | ||||||
|     } |  | ||||||
| } | } | ||||||
|   | |||||||
| @@ -6,44 +6,42 @@ | |||||||
|  |  | ||||||
| package dan200.computercraft.client.gui; | package dan200.computercraft.client.gui; | ||||||
|  |  | ||||||
|  | import com.mojang.blaze3d.platform.GlStateManager; | ||||||
| import dan200.computercraft.shared.peripheral.diskdrive.ContainerDiskDrive; | import dan200.computercraft.shared.peripheral.diskdrive.ContainerDiskDrive; | ||||||
| import net.minecraft.client.gui.inventory.GuiContainer; | import net.minecraft.client.gui.screen.inventory.ContainerScreen; | ||||||
| import net.minecraft.client.renderer.GlStateManager; | import net.minecraft.entity.player.PlayerInventory; | ||||||
| import net.minecraft.client.resources.I18n; |  | ||||||
| import net.minecraft.util.ResourceLocation; | import net.minecraft.util.ResourceLocation; | ||||||
|  | import net.minecraft.util.text.ITextComponent; | ||||||
|  |  | ||||||
| public class GuiDiskDrive extends GuiContainer | public class GuiDiskDrive extends ContainerScreen<ContainerDiskDrive> | ||||||
| { | { | ||||||
|     private static final ResourceLocation BACKGROUND = new ResourceLocation( "computercraft", "textures/gui/disk_drive.png" ); |     private static final ResourceLocation BACKGROUND = new ResourceLocation( "computercraft", "textures/gui/disk_drive.png" ); | ||||||
|  |  | ||||||
|     private final ContainerDiskDrive m_container; |     public GuiDiskDrive( ContainerDiskDrive container, PlayerInventory player, ITextComponent title ) | ||||||
|  |  | ||||||
|     public GuiDiskDrive( ContainerDiskDrive container ) |  | ||||||
|     { |     { | ||||||
|         super( container ); |         super( container, player, title ); | ||||||
|         m_container = container; |  | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @Override |     @Override | ||||||
|     protected void drawGuiContainerForegroundLayer( int mouseX, int mouseY ) |     protected void drawGuiContainerForegroundLayer( int mouseX, int mouseY ) | ||||||
|     { |     { | ||||||
|         String title = m_container.getDiskDrive().getDisplayName().getString(); |         String title = this.title.getFormattedText(); | ||||||
|         fontRenderer.drawString( title, (xSize - fontRenderer.getStringWidth( title )) / 2.0f, 6, 0x404040 ); |         font.drawString( title, (xSize - font.getStringWidth( title )) / 2.0f, 6, 0x404040 ); | ||||||
|         fontRenderer.drawString( I18n.format( "container.inventory" ), 8, ySize - 96 + 2, 0x404040 ); |         font.drawString( title, 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.color4f( 1.0F, 1.0F, 1.0F, 1.0F ); |         GlStateManager.color4f( 1.0F, 1.0F, 1.0F, 1.0F ); | ||||||
|         mc.getTextureManager().bindTexture( BACKGROUND ); |         minecraft.getTextureManager().bindTexture( BACKGROUND ); | ||||||
|         drawTexturedModalRect( guiLeft, guiTop, 0, 0, xSize, ySize ); |         blit( guiLeft, guiTop, 0, 0, xSize, ySize ); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @Override |     @Override | ||||||
|     public void render( int mouseX, int mouseY, float partialTicks ) |     public void render( int mouseX, int mouseY, float partialTicks ) | ||||||
|     { |     { | ||||||
|         drawDefaultBackground(); |         renderBackground(); | ||||||
|         super.render( mouseX, mouseY, partialTicks ); |         super.render( mouseX, mouseY, partialTicks ); | ||||||
|         renderHoveredToolTip( mouseX, mouseY ); |         renderHoveredToolTip( mouseX, mouseY ); | ||||||
|     } |     } | ||||||
|   | |||||||
| @@ -1,34 +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.computer.core.ComputerFamily; |  | ||||||
| import dan200.computercraft.shared.pocket.inventory.ContainerPocketComputer; |  | ||||||
| import dan200.computercraft.shared.pocket.items.ItemPocketComputer; |  | ||||||
| import net.minecraft.item.Item; |  | ||||||
| import net.minecraft.item.ItemStack; |  | ||||||
|  |  | ||||||
| public class GuiPocketComputer extends GuiComputer |  | ||||||
| { |  | ||||||
|     public GuiPocketComputer( ContainerPocketComputer container ) |  | ||||||
|     { |  | ||||||
|         super( |  | ||||||
|             container, |  | ||||||
|             getFamily( container.getStack() ), |  | ||||||
|             ItemPocketComputer.createClientComputer( container.getStack() ), |  | ||||||
|             ComputerCraft.terminalWidth_pocketComputer, |  | ||||||
|             ComputerCraft.terminalHeight_pocketComputer |  | ||||||
|         ); |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     private static ComputerFamily getFamily( ItemStack stack ) |  | ||||||
|     { |  | ||||||
|         Item item = stack.getItem(); |  | ||||||
|         return item instanceof ItemPocketComputer ? ((ItemPocketComputer) item).getFamily() : ComputerFamily.Normal; |  | ||||||
|     } |  | ||||||
| } |  | ||||||
| @@ -6,46 +6,45 @@ | |||||||
|  |  | ||||||
| package dan200.computercraft.client.gui; | package dan200.computercraft.client.gui; | ||||||
|  |  | ||||||
|  | import com.mojang.blaze3d.platform.GlStateManager; | ||||||
| import dan200.computercraft.shared.peripheral.printer.ContainerPrinter; | import dan200.computercraft.shared.peripheral.printer.ContainerPrinter; | ||||||
| import net.minecraft.client.gui.inventory.GuiContainer; | import net.minecraft.client.gui.screen.inventory.ContainerScreen; | ||||||
| import net.minecraft.client.renderer.GlStateManager; |  | ||||||
| import net.minecraft.client.resources.I18n; | import net.minecraft.client.resources.I18n; | ||||||
|  | import net.minecraft.entity.player.PlayerInventory; | ||||||
| import net.minecraft.util.ResourceLocation; | import net.minecraft.util.ResourceLocation; | ||||||
|  | import net.minecraft.util.text.ITextComponent; | ||||||
|  |  | ||||||
| public class GuiPrinter extends GuiContainer | public class GuiPrinter extends ContainerScreen<ContainerPrinter> | ||||||
| { | { | ||||||
|     private static final ResourceLocation BACKGROUND = new ResourceLocation( "computercraft", "textures/gui/printer.png" ); |     private static final ResourceLocation BACKGROUND = new ResourceLocation( "computercraft", "textures/gui/printer.png" ); | ||||||
|  |  | ||||||
|     private final ContainerPrinter container; |     public GuiPrinter( ContainerPrinter container, PlayerInventory player, ITextComponent title ) | ||||||
|  |  | ||||||
|     public GuiPrinter( ContainerPrinter container ) |  | ||||||
|     { |     { | ||||||
|         super( container ); |         super( container, player, title ); | ||||||
|         this.container = container; |  | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @Override |     @Override | ||||||
|     protected void drawGuiContainerForegroundLayer( int mouseX, int mouseY ) |     protected void drawGuiContainerForegroundLayer( int mouseX, int mouseY ) | ||||||
|     { |     { | ||||||
|         String title = container.getPrinter().getDisplayName().getString(); |         String title = getTitle().getFormattedText(); | ||||||
|         fontRenderer.drawString( title, (xSize - fontRenderer.getStringWidth( title )) / 2.0f, 6, 0x404040 ); |         font.drawString( title, (xSize - font.getStringWidth( title )) / 2.0f, 6, 0x404040 ); | ||||||
|         fontRenderer.drawString( I18n.format( "container.inventory" ), 8, ySize - 96 + 2, 0x404040 ); |         font.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.color4f( 1.0F, 1.0F, 1.0F, 1.0F ); |         GlStateManager.color4f( 1.0F, 1.0F, 1.0F, 1.0F ); | ||||||
|         mc.getTextureManager().bindTexture( BACKGROUND ); |         minecraft.getTextureManager().bindTexture( BACKGROUND ); | ||||||
|         drawTexturedModalRect( guiLeft, guiTop, 0, 0, xSize, ySize ); |         blit( guiLeft, guiTop, 0, 0, xSize, ySize ); | ||||||
|  |  | ||||||
|         if( container.isPrinting() ) drawTexturedModalRect( guiLeft + 34, guiTop + 21, 176, 0, 25, 45 ); |         if( getContainer().isPrinting() ) blit( guiLeft + 34, guiTop + 21, 176, 0, 25, 45 ); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @Override |     @Override | ||||||
|     public void render( int mouseX, int mouseY, float partialTicks ) |     public void render( int mouseX, int mouseY, float partialTicks ) | ||||||
|     { |     { | ||||||
|         drawDefaultBackground(); |         renderBackground(); | ||||||
|         super.render( mouseX, mouseY, partialTicks ); |         super.render( mouseX, mouseY, partialTicks ); | ||||||
|         renderHoveredToolTip( mouseX, mouseY ); |         renderHoveredToolTip( mouseX, mouseY ); | ||||||
|     } |     } | ||||||
|   | |||||||
| @@ -6,16 +6,18 @@ | |||||||
|  |  | ||||||
| package dan200.computercraft.client.gui; | package dan200.computercraft.client.gui; | ||||||
|  |  | ||||||
|  | import com.mojang.blaze3d.platform.GlStateManager; | ||||||
| import dan200.computercraft.core.terminal.TextBuffer; | import dan200.computercraft.core.terminal.TextBuffer; | ||||||
| import dan200.computercraft.shared.common.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.screen.inventory.ContainerScreen; | ||||||
| import net.minecraft.client.renderer.GlStateManager; | import net.minecraft.entity.player.PlayerInventory; | ||||||
|  | import net.minecraft.util.text.ITextComponent; | ||||||
| import org.lwjgl.glfw.GLFW; | import org.lwjgl.glfw.GLFW; | ||||||
|  |  | ||||||
| import static dan200.computercraft.client.render.PrintoutRenderer.*; | import static dan200.computercraft.client.render.PrintoutRenderer.*; | ||||||
|  |  | ||||||
| public class GuiPrintout extends GuiContainer | public class GuiPrintout extends ContainerScreen<ContainerHeldItem> | ||||||
| { | { | ||||||
|     private final boolean m_book; |     private final boolean m_book; | ||||||
|     private final int m_pages; |     private final int m_pages; | ||||||
| @@ -23,9 +25,9 @@ public class GuiPrintout extends GuiContainer | |||||||
|     private final TextBuffer[] m_colours; |     private final TextBuffer[] m_colours; | ||||||
|     private int m_page; |     private int m_page; | ||||||
|  |  | ||||||
|     public GuiPrintout( ContainerHeldItem container ) |     public GuiPrintout( ContainerHeldItem container, PlayerInventory player, ITextComponent title ) | ||||||
|     { |     { | ||||||
|         super( container ); |         super( container, player, title ); | ||||||
|  |  | ||||||
|         ySize = Y_SIZE; |         ySize = Y_SIZE; | ||||||
|  |  | ||||||
| @@ -63,9 +65,9 @@ public class GuiPrintout extends GuiContainer | |||||||
|     } |     } | ||||||
|  |  | ||||||
|     @Override |     @Override | ||||||
|     public boolean mouseScrolled( double delta ) |     public boolean mouseScrolled( double x, double y, double delta ) | ||||||
|     { |     { | ||||||
|         if( super.mouseScrolled( delta ) ) return true; |         if( super.mouseScrolled( x, y, delta ) ) return true; | ||||||
|         if( delta < 0 ) |         if( delta < 0 ) | ||||||
|         { |         { | ||||||
|             // Scroll up goes to the next page |             // Scroll up goes to the next page | ||||||
| @@ -90,7 +92,7 @@ public class GuiPrintout extends GuiContainer | |||||||
|         GlStateManager.color4f( 1.0f, 1.0f, 1.0f, 1.0f ); |         GlStateManager.color4f( 1.0f, 1.0f, 1.0f, 1.0f ); | ||||||
|         GlStateManager.enableDepthTest(); |         GlStateManager.enableDepthTest(); | ||||||
|  |  | ||||||
|         drawBorder( guiLeft, guiTop, zLevel, m_page, m_pages, m_book ); |         drawBorder( guiLeft, guiTop, blitOffset, 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 ); | ||||||
|     } |     } | ||||||
|  |  | ||||||
| @@ -98,9 +100,9 @@ public class GuiPrintout extends GuiContainer | |||||||
|     public void render( 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--; |         blitOffset--; | ||||||
|         drawDefaultBackground(); |         renderBackground(); | ||||||
|         zLevel++; |         blitOffset++; | ||||||
|  |  | ||||||
|         super.render( mouseX, mouseY, partialTicks ); |         super.render( mouseX, mouseY, partialTicks ); | ||||||
|         renderHoveredToolTip( mouseX, mouseY ); |         renderHoveredToolTip( mouseX, mouseY ); | ||||||
|   | |||||||
| @@ -6,18 +6,20 @@ | |||||||
|  |  | ||||||
| package dan200.computercraft.client.gui; | package dan200.computercraft.client.gui; | ||||||
|  |  | ||||||
|  | import com.mojang.blaze3d.platform.GlStateManager; | ||||||
| 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.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.inventory.ContainerTurtle; | import dan200.computercraft.shared.turtle.inventory.ContainerTurtle; | ||||||
| import net.minecraft.client.gui.inventory.GuiContainer; | import net.minecraft.client.gui.screen.inventory.ContainerScreen; | ||||||
| import net.minecraft.client.renderer.GlStateManager; | import net.minecraft.entity.player.PlayerInventory; | ||||||
| import net.minecraft.util.ResourceLocation; | import net.minecraft.util.ResourceLocation; | ||||||
|  | import net.minecraft.util.text.ITextComponent; | ||||||
|  | import org.lwjgl.glfw.GLFW; | ||||||
|  |  | ||||||
| public class GuiTurtle extends GuiContainer | public class GuiTurtle extends ContainerScreen<ContainerTurtle> | ||||||
| { | { | ||||||
|     private static final ResourceLocation BACKGROUND_NORMAL = new ResourceLocation( "computercraft", "textures/gui/turtle_normal.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" ); | ||||||
| @@ -30,29 +32,29 @@ public class GuiTurtle extends GuiContainer | |||||||
|     private WidgetTerminal terminal; |     private WidgetTerminal terminal; | ||||||
|     private WidgetWrapper terminalWrapper; |     private WidgetWrapper terminalWrapper; | ||||||
|  |  | ||||||
|     public GuiTurtle( TileTurtle turtle, ContainerTurtle container ) |     public GuiTurtle( ContainerTurtle container, PlayerInventory player, ITextComponent title ) | ||||||
|     { |     { | ||||||
|         super( container ); |         super( container, player, title ); | ||||||
|  |  | ||||||
|         m_container = container; |         m_container = container; | ||||||
|         m_family = turtle.getFamily(); |         m_family = container.getFamily(); | ||||||
|         m_computer = turtle.getClientComputer(); |         m_computer = (ClientComputer) container.getComputer(); | ||||||
|  |  | ||||||
|         xSize = 254; |         xSize = 254; | ||||||
|         ySize = 217; |         ySize = 217; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @Override |     @Override | ||||||
|     protected void initGui() |     protected void init() | ||||||
|     { |     { | ||||||
|         super.initGui(); |         super.init(); | ||||||
|         mc.keyboardListener.enableRepeatEvents( true ); |         minecraft.keyboardListener.enableRepeatEvents( true ); | ||||||
|  |  | ||||||
|         int termPxWidth = ComputerCraft.terminalWidth_turtle * FixedWidthFontRenderer.FONT_WIDTH; |         int termPxWidth = ComputerCraft.terminalWidth_turtle * FixedWidthFontRenderer.FONT_WIDTH; | ||||||
|         int termPxHeight = ComputerCraft.terminalHeight_turtle * FixedWidthFontRenderer.FONT_HEIGHT; |         int termPxHeight = ComputerCraft.terminalHeight_turtle * FixedWidthFontRenderer.FONT_HEIGHT; | ||||||
|  |  | ||||||
|         terminal = new WidgetTerminal( |         terminal = new WidgetTerminal( | ||||||
|             mc, () -> m_computer, |             minecraft, () -> m_computer, | ||||||
|             ComputerCraft.terminalWidth_turtle, |             ComputerCraft.terminalWidth_turtle, | ||||||
|             ComputerCraft.terminalHeight_turtle, |             ComputerCraft.terminalHeight_turtle, | ||||||
|             2, 2, 2, 2 |             2, 2, 2, 2 | ||||||
| @@ -64,11 +66,12 @@ public class GuiTurtle extends GuiContainer | |||||||
|     } |     } | ||||||
|  |  | ||||||
|     @Override |     @Override | ||||||
|     public void onGuiClosed() |     public void removed() | ||||||
|     { |     { | ||||||
|  |         super.removed(); | ||||||
|         children.remove( terminal ); |         children.remove( terminal ); | ||||||
|         terminal = null; |         terminal = null; | ||||||
|         mc.keyboardListener.enableRepeatEvents( false ); |         minecraft.keyboardListener.enableRepeatEvents( false ); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @Override |     @Override | ||||||
| @@ -78,6 +81,18 @@ public class GuiTurtle extends GuiContainer | |||||||
|         terminal.update(); |         terminal.update(); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     @Override | ||||||
|  |     public boolean keyPressed( int key, int scancode, int modifiers ) | ||||||
|  |     { | ||||||
|  |         // Forward the tab key to the terminal, rather than moving between controls. | ||||||
|  |         if( key == GLFW.GLFW_KEY_TAB && getFocused() != null && getFocused() == terminalWrapper ) | ||||||
|  |         { | ||||||
|  |             return getFocused().keyPressed( key, scancode, modifiers ); | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         return super.keyPressed( key, scancode, modifiers ); | ||||||
|  |     } | ||||||
|  |  | ||||||
|     private void drawSelectionSlot( boolean advanced ) |     private void drawSelectionSlot( boolean advanced ) | ||||||
|     { |     { | ||||||
|         // Draw selection slot |         // Draw selection slot | ||||||
| @@ -87,8 +102,8 @@ public class GuiTurtle extends GuiContainer | |||||||
|             GlStateManager.color4f( 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 ); |             minecraft.getTextureManager().bindTexture( advanced ? BACKGROUND_ADVANCED : BACKGROUND_NORMAL ); | ||||||
|             drawTexturedModalRect( guiLeft + m_container.m_turtleInvStartX - 2 + slotX * 18, guiTop + m_container.m_playerInvStartY - 2 + slotY * 18, 0, 217, 24, 24 ); |             blit( guiLeft + ContainerTurtle.TURTLE_START_X - 2 + slotX * 18, guiTop + ContainerTurtle.PLAYER_START_Y - 2 + slotY * 18, 0, 217, 24, 24 ); | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  |  | ||||||
| @@ -101,8 +116,8 @@ public class GuiTurtle extends GuiContainer | |||||||
|  |  | ||||||
|         // Draw border/inventory |         // Draw border/inventory | ||||||
|         GlStateManager.color4f( 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 ); |         minecraft.getTextureManager().bindTexture( advanced ? BACKGROUND_ADVANCED : BACKGROUND_NORMAL ); | ||||||
|         drawTexturedModalRect( guiLeft, guiTop, 0, 0, xSize, ySize ); |         blit( guiLeft, guiTop, 0, 0, xSize, ySize ); | ||||||
|  |  | ||||||
|         drawSelectionSlot( advanced ); |         drawSelectionSlot( advanced ); | ||||||
|     } |     } | ||||||
| @@ -110,8 +125,15 @@ public class GuiTurtle extends GuiContainer | |||||||
|     @Override |     @Override | ||||||
|     public void render( int mouseX, int mouseY, float partialTicks ) |     public void render( int mouseX, int mouseY, float partialTicks ) | ||||||
|     { |     { | ||||||
|         drawDefaultBackground(); |         renderBackground(); | ||||||
|         super.render( mouseX, mouseY, partialTicks ); |         super.render( mouseX, mouseY, partialTicks ); | ||||||
|         renderHoveredToolTip( mouseX, mouseY ); |         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 ); | ||||||
|  |     } | ||||||
| } | } | ||||||
|   | |||||||
| @@ -6,6 +6,7 @@ | |||||||
|  |  | ||||||
| package dan200.computercraft.client.gui.widgets; | package dan200.computercraft.client.gui.widgets; | ||||||
|  |  | ||||||
|  | import com.mojang.blaze3d.platform.GlStateManager; | ||||||
| import dan200.computercraft.client.FrameInfo; | import dan200.computercraft.client.FrameInfo; | ||||||
| import dan200.computercraft.client.gui.FixedWidthFontRenderer; | import dan200.computercraft.client.gui.FixedWidthFontRenderer; | ||||||
| import dan200.computercraft.core.terminal.Terminal; | import dan200.computercraft.core.terminal.Terminal; | ||||||
| @@ -17,7 +18,6 @@ import dan200.computercraft.shared.util.Palette; | |||||||
| import net.minecraft.client.Minecraft; | import net.minecraft.client.Minecraft; | ||||||
| import net.minecraft.client.gui.IGuiEventListener; | import net.minecraft.client.gui.IGuiEventListener; | ||||||
| import net.minecraft.client.renderer.BufferBuilder; | import net.minecraft.client.renderer.BufferBuilder; | ||||||
| import net.minecraft.client.renderer.GlStateManager; |  | ||||||
| import net.minecraft.client.renderer.Tessellator; | import net.minecraft.client.renderer.Tessellator; | ||||||
| import net.minecraft.client.renderer.vertex.DefaultVertexFormats; | import net.minecraft.client.renderer.vertex.DefaultVertexFormats; | ||||||
| import net.minecraft.util.SharedConstants; | import net.minecraft.util.SharedConstants; | ||||||
| @@ -35,6 +35,8 @@ public class WidgetTerminal implements IGuiEventListener | |||||||
|  |  | ||||||
|     private final Minecraft client; |     private final Minecraft client; | ||||||
|  |  | ||||||
|  |     private boolean focused; | ||||||
|  |  | ||||||
|     private final Supplier<ClientComputer> computer; |     private final Supplier<ClientComputer> computer; | ||||||
|     private final int termWidth; |     private final int termWidth; | ||||||
|     private final int termHeight; |     private final int termHeight; | ||||||
| @@ -239,25 +241,35 @@ public class WidgetTerminal implements IGuiEventListener | |||||||
|             charX = Math.min( Math.max( charX, 0 ), term.getWidth() - 1 ); |             charX = Math.min( Math.max( charX, 0 ), term.getWidth() - 1 ); | ||||||
|             charY = Math.min( Math.max( charY, 0 ), term.getHeight() - 1 ); |             charY = Math.min( Math.max( charY, 0 ), term.getHeight() - 1 ); | ||||||
|  |  | ||||||
|             computer.mouseDrag( button + 1, charX + 1, charY + 1 ); |             if( button == lastMouseButton && (charX != lastMouseX || charY != lastMouseY) ) | ||||||
|  |             { | ||||||
|             lastMouseX = charX; |                 computer.mouseDrag( button + 1, charX + 1, charY + 1 ); | ||||||
|             lastMouseY = charY; |                 lastMouseX = charX; | ||||||
|             lastMouseButton = button; |                 lastMouseY = charY; | ||||||
|  |             } | ||||||
|         } |         } | ||||||
|  |  | ||||||
|         return false; |         return false; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @Override |     @Override | ||||||
|     public boolean mouseScrolled( double delta ) |     public boolean mouseScrolled( double mouseX, double mouseY, double delta ) | ||||||
|     { |     { | ||||||
|         ClientComputer computer = this.computer.get(); |         ClientComputer computer = this.computer.get(); | ||||||
|         if( computer == null || !computer.isColour() ) return false; |         if( computer == null || !computer.isColour() || delta == 0 ) return false; | ||||||
|  |  | ||||||
|         if( lastMouseX >= 0 && lastMouseY >= 0 && delta != 0 ) |         Terminal term = computer.getTerminal(); | ||||||
|  |         if( term != null ) | ||||||
|         { |         { | ||||||
|             queueEvent( "mouse_scroll", delta < 0 ? 1 : -1, lastMouseX + 1, lastMouseY + 1 ); |             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.mouseScroll( delta < 0 ? 1 : -1, charX + 1, charY + 1 ); | ||||||
|  |  | ||||||
|  |             lastMouseX = charX; | ||||||
|  |             lastMouseY = charY; | ||||||
|         } |         } | ||||||
|  |  | ||||||
|         return true; |         return true; | ||||||
| @@ -284,9 +296,9 @@ public class WidgetTerminal implements IGuiEventListener | |||||||
|     } |     } | ||||||
|  |  | ||||||
|     @Override |     @Override | ||||||
|     public void focusChanged( boolean focused ) |     public boolean changeFocus( boolean reversed ) | ||||||
|     { |     { | ||||||
|         if( !focused ) |         if( focused ) | ||||||
|         { |         { | ||||||
|             // When blurring, we should make all keys go up |             // When blurring, we should make all keys go up | ||||||
|             for( int key = 0; key < keysDown.size(); key++ ) |             for( int key = 0; key < keysDown.size(); key++ ) | ||||||
| @@ -305,6 +317,8 @@ public class WidgetTerminal implements IGuiEventListener | |||||||
|  |  | ||||||
|             shutdownTimer = terminateTimer = rebootTimer = -1; |             shutdownTimer = terminateTimer = rebootTimer = -1; | ||||||
|         } |         } | ||||||
|  |         focused = !focused; | ||||||
|  |         return true; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     public void draw( int originX, int originY ) |     public void draw( int originX, int originY ) | ||||||
| @@ -414,4 +428,10 @@ public class WidgetTerminal implements IGuiEventListener | |||||||
|         ClientComputer computer = this.computer.get(); |         ClientComputer computer = this.computer.get(); | ||||||
|         if( computer != null ) computer.queueEvent( event, args ); |         if( computer != null ) computer.queueEvent( event, args ); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     @Override | ||||||
|  |     public boolean isMouseOver( double x, double y ) | ||||||
|  |     { | ||||||
|  |         return true; | ||||||
|  |     } | ||||||
| } | } | ||||||
|   | |||||||
| @@ -26,15 +26,9 @@ public class WidgetWrapper implements IGuiEventListener | |||||||
|     } |     } | ||||||
|  |  | ||||||
|     @Override |     @Override | ||||||
|     public void focusChanged( boolean b ) |     public boolean changeFocus( boolean b ) | ||||||
|     { |     { | ||||||
|         listener.focusChanged( b ); |         return listener.changeFocus( b ); | ||||||
|     } |  | ||||||
|  |  | ||||||
|     @Override |  | ||||||
|     public boolean canFocus() |  | ||||||
|     { |  | ||||||
|         return listener.canFocus(); |  | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @Override |     @Override | ||||||
| @@ -59,9 +53,9 @@ public class WidgetWrapper implements IGuiEventListener | |||||||
|     } |     } | ||||||
|  |  | ||||||
|     @Override |     @Override | ||||||
|     public boolean mouseScrolled( double delta ) |     public boolean mouseScrolled( double x, double y, double delta ) | ||||||
|     { |     { | ||||||
|         return listener.mouseScrolled( delta ); |         return listener.mouseScrolled( x, y, delta ); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @Override |     @Override | ||||||
| @@ -101,4 +95,11 @@ public class WidgetWrapper implements IGuiEventListener | |||||||
|     { |     { | ||||||
|         return height; |         return height; | ||||||
|     } |     } | ||||||
|  |  | ||||||
|  |     @Override | ||||||
|  |     public boolean isMouseOver( double x, double y ) | ||||||
|  |     { | ||||||
|  |         double dx = x - this.x, dy = y - this.y; | ||||||
|  |         return dx >= 0 && dx < width && dy >= 0 && dy < height; | ||||||
|  |     } | ||||||
| } | } | ||||||
|   | |||||||
| @@ -11,29 +11,25 @@ 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.computer.blocks.TileComputer; | import dan200.computercraft.shared.common.ContainerHeldItem; | ||||||
| import dan200.computercraft.shared.computer.core.ClientComputer; | import dan200.computercraft.shared.computer.inventory.ContainerComputer; | ||||||
| import dan200.computercraft.shared.computer.inventory.ContainerViewComputer; | import dan200.computercraft.shared.computer.inventory.ContainerViewComputer; | ||||||
| import dan200.computercraft.shared.network.container.*; | import dan200.computercraft.shared.peripheral.diskdrive.ContainerDiskDrive; | ||||||
| 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.peripheral.printer.ContainerPrinter; | ||||||
|  | import dan200.computercraft.shared.pocket.inventory.ContainerPocketComputer; | ||||||
| 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.ScreenManager; | ||||||
| import net.minecraft.client.gui.inventory.GuiContainer; |  | ||||||
| import net.minecraft.entity.player.EntityPlayer; |  | ||||||
| import net.minecraftforge.api.distmarker.Dist; | 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.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.event.lifecycle.FMLClientSetupEvent; | import net.minecraftforge.fml.event.lifecycle.FMLClientSetupEvent; | ||||||
|  |  | ||||||
| import java.util.function.BiFunction; |  | ||||||
|  |  | ||||||
| @Mod.EventBusSubscriber( modid = ComputerCraft.MOD_ID, value = Dist.CLIENT, bus = Mod.EventBusSubscriber.Bus.MOD ) | @Mod.EventBusSubscriber( modid = ComputerCraft.MOD_ID, value = Dist.CLIENT, bus = Mod.EventBusSubscriber.Bus.MOD ) | ||||||
| public final class ComputerCraftProxyClient | public final class ComputerCraftProxyClient | ||||||
| { | { | ||||||
| @@ -50,35 +46,17 @@ public final class ComputerCraftProxyClient | |||||||
|  |  | ||||||
|     private static void registerContainers() |     private static void registerContainers() | ||||||
|     { |     { | ||||||
|         ContainerType.registerGui( TileEntityContainerType::computer, ( packet, player ) -> |         // My IDE doesn't think so, but we do actually need these generics. | ||||||
|             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 ); |         ScreenManager.<ContainerComputer, GuiComputer<ContainerComputer>>registerFactory( ContainerComputer.TYPE, GuiComputer::create ); | ||||||
|         ContainerType.registerGui( PrintoutContainerType::new, GuiPrintout::new ); |         ScreenManager.<ContainerPocketComputer, GuiComputer<ContainerPocketComputer>>registerFactory( ContainerPocketComputer.TYPE, GuiComputer::createPocket ); | ||||||
|         ContainerType.registerGui( ViewComputerContainerType::new, ( packet, player ) -> { |         ScreenManager.registerFactory( ContainerTurtle.TYPE, GuiTurtle::new ); | ||||||
|             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 ); |         ScreenManager.registerFactory( ContainerPrinter.TYPE, GuiPrinter::new ); | ||||||
|             return new GuiComputer( container, packet.family, computer, packet.width, packet.height ); |         ScreenManager.registerFactory( ContainerDiskDrive.TYPE, GuiDiskDrive::new ); | ||||||
|         } ); |         ScreenManager.registerFactory( ContainerHeldItem.PRINTOUT_TYPE, GuiPrintout::new ); | ||||||
|  |  | ||||||
|         ModLoadingContext.get().registerExtensionPoint( ExtensionPoint.GUIFACTORY, () -> packet -> { |         ScreenManager.<ContainerViewComputer, GuiComputer<ContainerViewComputer>>registerFactory( ContainerViewComputer.TYPE, GuiComputer::createView ); | ||||||
|             ContainerType<?> type = ContainerType.factories.get( packet.getId() ).get(); |  | ||||||
|             if( packet.getAdditionalData() != null ) type.fromBytes( packet.getAdditionalData() ); |  | ||||||
|             @SuppressWarnings( "unchecked" ) |  | ||||||
|             BiFunction<ContainerType<?>, EntityPlayer, GuiContainer> factory = (BiFunction<ContainerType<?>, EntityPlayer, GuiContainer>) ContainerType.guiFactories.get( packet.getId() ); |  | ||||||
|             return factory.apply( type, Minecraft.getInstance().player ); |  | ||||||
|         } ); |  | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @Mod.EventBusSubscriber( modid = ComputerCraft.MOD_ID, value = Dist.CLIENT ) |     @Mod.EventBusSubscriber( modid = ComputerCraft.MOD_ID, value = Dist.CLIENT ) | ||||||
|   | |||||||
| @@ -6,17 +6,19 @@ | |||||||
|  |  | ||||||
| package dan200.computercraft.client.render; | package dan200.computercraft.client.render; | ||||||
|  |  | ||||||
|  | import com.mojang.blaze3d.platform.GlStateManager; | ||||||
| import dan200.computercraft.ComputerCraft; | import dan200.computercraft.ComputerCraft; | ||||||
| import dan200.computercraft.shared.peripheral.modem.wired.BlockCable; | import dan200.computercraft.shared.peripheral.modem.wired.BlockCable; | ||||||
| import dan200.computercraft.shared.peripheral.modem.wired.CableShapes; | 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.BlockState; | ||||||
| import net.minecraft.client.Minecraft; | import net.minecraft.client.Minecraft; | ||||||
| import net.minecraft.client.renderer.GlStateManager; | import net.minecraft.client.renderer.ActiveRenderInfo; | ||||||
| import net.minecraft.client.renderer.WorldRenderer; | import net.minecraft.client.renderer.WorldRenderer; | ||||||
| import net.minecraft.entity.player.EntityPlayer; |  | ||||||
| import net.minecraft.util.math.BlockPos; | import net.minecraft.util.math.BlockPos; | ||||||
|  | import net.minecraft.util.math.BlockRayTraceResult; | ||||||
| import net.minecraft.util.math.RayTraceResult; | import net.minecraft.util.math.RayTraceResult; | ||||||
|  | import net.minecraft.util.math.Vec3d; | ||||||
| import net.minecraft.util.math.shapes.VoxelShape; | 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.api.distmarker.Dist; | ||||||
| @@ -36,17 +38,19 @@ public final class CableHighlightRenderer | |||||||
|      * Draw an outline for a specific part of a cable "Multipart". |      * Draw an outline for a specific part of a cable "Multipart". | ||||||
|      * |      * | ||||||
|      * @param event The event to observe |      * @param event The event to observe | ||||||
|      * @see WorldRenderer#drawSelectionBox(EntityPlayer, RayTraceResult, int, float) |      * @see WorldRenderer#drawSelectionBox(ActiveRenderInfo, RayTraceResult, int) | ||||||
|      */ |      */ | ||||||
|     @SubscribeEvent |     @SubscribeEvent | ||||||
|     public static void drawHighlight( DrawBlockHighlightEvent event ) |     public static void drawHighlight( DrawBlockHighlightEvent event ) | ||||||
|     { |     { | ||||||
|         if( event.getTarget().type != RayTraceResult.Type.BLOCK ) return; |         if( event.getTarget().getType() != RayTraceResult.Type.BLOCK ) return; | ||||||
|  |  | ||||||
|         BlockPos pos = event.getTarget().getBlockPos(); |         BlockRayTraceResult hit = (BlockRayTraceResult) event.getTarget(); | ||||||
|         World world = event.getPlayer().getEntityWorld(); |         BlockPos pos = hit.getPos(); | ||||||
|  |         World world = event.getInfo().getRenderViewEntity().getEntityWorld(); | ||||||
|  |         ActiveRenderInfo info = event.getInfo(); | ||||||
|  |  | ||||||
|         IBlockState state = world.getBlockState( pos ); |         BlockState state = world.getBlockState( pos ); | ||||||
|  |  | ||||||
|         // We only care about instances with both cable and modem. |         // 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 ) ) |         if( state.getBlock() != ComputerCraft.Blocks.cable || state.get( BlockCable.MODEM ).getFacing() == null || !state.get( BlockCable.CABLE ) ) | ||||||
| @@ -56,33 +60,31 @@ public final class CableHighlightRenderer | |||||||
|  |  | ||||||
|         event.setCanceled( true ); |         event.setCanceled( true ); | ||||||
|  |  | ||||||
|         EntityPlayer player = event.getPlayer(); |  | ||||||
|         Minecraft mc = Minecraft.getInstance(); |         Minecraft mc = Minecraft.getInstance(); | ||||||
|         float partialTicks = event.getPartialTicks(); |  | ||||||
|  |  | ||||||
|         GlStateManager.enableBlend(); |         GlStateManager.enableBlend(); | ||||||
|         GlStateManager.blendFuncSeparate( GlStateManager.SourceFactor.SRC_ALPHA, GlStateManager.DestFactor.ONE_MINUS_SRC_ALPHA, GlStateManager.SourceFactor.ONE, GlStateManager.DestFactor.ZERO ); |         GlStateManager.blendFuncSeparate( GlStateManager.SourceFactor.SRC_ALPHA, GlStateManager.DestFactor.ONE_MINUS_SRC_ALPHA, GlStateManager.SourceFactor.ONE, GlStateManager.DestFactor.ZERO ); | ||||||
|         GlStateManager.lineWidth( Math.max( 2.5F, mc.mainWindow.getFramebufferWidth() / 1920.0F * 2.5F ) ); |         GlStateManager.lineWidth( Math.max( 2.5F, mc.mainWindow.getFramebufferWidth() / 1920.0F * 2.5F ) ); | ||||||
|         GlStateManager.disableTexture2D(); |         GlStateManager.disableTexture(); | ||||||
|         GlStateManager.depthMask( false ); |         GlStateManager.depthMask( false ); | ||||||
|         GlStateManager.matrixMode( GL11.GL_PROJECTION ); |         GlStateManager.matrixMode( GL11.GL_PROJECTION ); | ||||||
|         GlStateManager.pushMatrix(); |         GlStateManager.pushMatrix(); | ||||||
|         GlStateManager.scalef( 1.0F, 1.0F, 0.999F ); |         GlStateManager.scalef( 1.0F, 1.0F, 0.999F ); | ||||||
|  |  | ||||||
|         double x = player.lastTickPosX + (player.posX - player.lastTickPosX) * partialTicks; |         VoxelShape shape = WorldUtil.isVecInside( CableShapes.getModemShape( state ), hit.getHitVec().subtract( pos.getX(), pos.getY(), pos.getZ() ) ) | ||||||
|         double y = player.lastTickPosY + (player.posY - player.lastTickPosY) * partialTicks; |  | ||||||
|         double z = player.lastTickPosZ + (player.posZ - player.lastTickPosZ) * partialTicks; |  | ||||||
|  |  | ||||||
|         VoxelShape shape = WorldUtil.isVecInside( CableShapes.getModemShape( state ), event.getTarget().hitVec.subtract( pos.getX(), pos.getY(), pos.getZ() ) ) |  | ||||||
|             ? CableShapes.getModemShape( state ) |             ? CableShapes.getModemShape( state ) | ||||||
|             : CableShapes.getCableShape( state ); |             : CableShapes.getCableShape( state ); | ||||||
|  |  | ||||||
|         WorldRenderer.drawShape( shape, pos.getX() - x, pos.getY() - y, pos.getZ() - z, 0.0F, 0.0F, 0.0F, 0.4F ); |         Vec3d cameraPos = info.getProjectedView(); | ||||||
|  |         WorldRenderer.drawShape( | ||||||
|  |             shape, pos.getX() - cameraPos.getX(), pos.getY() - cameraPos.getY(), pos.getZ() - cameraPos.getZ(), | ||||||
|  |             0.0F, 0.0F, 0.0F, 0.4F | ||||||
|  |         ); | ||||||
|  |  | ||||||
|         GlStateManager.popMatrix(); |         GlStateManager.popMatrix(); | ||||||
|         GlStateManager.matrixMode( GL11.GL_MODELVIEW ); |         GlStateManager.matrixMode( GL11.GL_MODELVIEW ); | ||||||
|         GlStateManager.depthMask( true ); |         GlStateManager.depthMask( true ); | ||||||
|         GlStateManager.enableTexture2D(); |         GlStateManager.enableTexture(); | ||||||
|         GlStateManager.disableBlend(); |         GlStateManager.disableBlend(); | ||||||
|     } |     } | ||||||
| } | } | ||||||
|   | |||||||
| @@ -6,38 +6,38 @@ | |||||||
|  |  | ||||||
| package dan200.computercraft.client.render; | package dan200.computercraft.client.render; | ||||||
|  |  | ||||||
|  | import com.mojang.blaze3d.platform.GlStateManager; | ||||||
| import net.minecraft.client.Minecraft; | import net.minecraft.client.Minecraft; | ||||||
| import net.minecraft.client.renderer.FirstPersonRenderer; | import net.minecraft.client.renderer.FirstPersonRenderer; | ||||||
| import net.minecraft.client.renderer.GlStateManager; | import net.minecraft.entity.player.PlayerEntity; | ||||||
| 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.Hand; | ||||||
| import net.minecraft.util.EnumHandSide; | import net.minecraft.util.HandSide; | ||||||
| import net.minecraft.util.math.MathHelper; | import net.minecraft.util.math.MathHelper; | ||||||
|  |  | ||||||
| public abstract class ItemMapLikeRenderer | 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 FirstPersonRenderer#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( Hand hand, float pitch, float equipProgress, float swingProgress, ItemStack stack ) | ||||||
|     { |     { | ||||||
|         EntityPlayer player = Minecraft.getInstance().player; |         PlayerEntity player = Minecraft.getInstance().player; | ||||||
|  |  | ||||||
|         GlStateManager.pushMatrix(); |         GlStateManager.pushMatrix(); | ||||||
|         if( hand == EnumHand.MAIN_HAND && player.getHeldItemOffhand().isEmpty() ) |         if( hand == Hand.MAIN_HAND && player.getHeldItemOffhand().isEmpty() ) | ||||||
|         { |         { | ||||||
|             renderItemFirstPersonCenter( pitch, equipProgress, swingProgress, stack ); |             renderItemFirstPersonCenter( pitch, equipProgress, swingProgress, stack ); | ||||||
|         } |         } | ||||||
|         else |         else | ||||||
|         { |         { | ||||||
|             renderItemFirstPersonSide( |             renderItemFirstPersonSide( | ||||||
|                 hand == EnumHand.MAIN_HAND ? player.getPrimaryHand() : player.getPrimaryHand().opposite(), |                 hand == Hand.MAIN_HAND ? player.getPrimaryHand() : player.getPrimaryHand().opposite(), | ||||||
|                 equipProgress, swingProgress, stack |                 equipProgress, swingProgress, stack | ||||||
|             ); |             ); | ||||||
|         } |         } | ||||||
| @@ -51,12 +51,12 @@ 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 FirstPersonRenderer#renderMapFirstPersonSide(float, EnumHandSide, float, ItemStack) |      * @see FirstPersonRenderer#renderMapFirstPersonSide(float, HandSide, float, ItemStack) | ||||||
|      */ |      */ | ||||||
|     private void renderItemFirstPersonSide( EnumHandSide side, float equipProgress, float swingProgress, ItemStack stack ) |     private void renderItemFirstPersonSide( HandSide side, float equipProgress, float swingProgress, ItemStack stack ) | ||||||
|     { |     { | ||||||
|         Minecraft minecraft = Minecraft.getInstance(); |         Minecraft minecraft = Minecraft.getInstance(); | ||||||
|         float offset = side == EnumHandSide.RIGHT ? 1f : -1f; |         float offset = side == HandSide.RIGHT ? 1f : -1f; | ||||||
|         GlStateManager.translatef( 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 | ||||||
| @@ -87,7 +87,7 @@ public abstract class ItemMapLikeRenderer | |||||||
|     } |     } | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * Render an item in the middle of the screen |      * Render an item in the middle of the screen. | ||||||
|      * |      * | ||||||
|      * @param pitch         The pitch of the player |      * @param pitch         The pitch of the player | ||||||
|      * @param equipProgress The equip progress of this item |      * @param equipProgress The equip progress of this item | ||||||
|   | |||||||
| @@ -6,6 +6,7 @@ | |||||||
|  |  | ||||||
| package dan200.computercraft.client.render; | package dan200.computercraft.client.render; | ||||||
|  |  | ||||||
|  | import com.mojang.blaze3d.platform.GlStateManager; | ||||||
| import dan200.computercraft.ComputerCraft; | import dan200.computercraft.ComputerCraft; | ||||||
| import dan200.computercraft.client.FrameInfo; | import dan200.computercraft.client.FrameInfo; | ||||||
| import dan200.computercraft.client.gui.FixedWidthFontRenderer; | import dan200.computercraft.client.gui.FixedWidthFontRenderer; | ||||||
| @@ -18,7 +19,6 @@ import dan200.computercraft.shared.util.Colour; | |||||||
| import dan200.computercraft.shared.util.Palette; | import dan200.computercraft.shared.util.Palette; | ||||||
| import net.minecraft.client.Minecraft; | import net.minecraft.client.Minecraft; | ||||||
| import net.minecraft.client.renderer.BufferBuilder; | import net.minecraft.client.renderer.BufferBuilder; | ||||||
| import net.minecraft.client.renderer.GlStateManager; |  | ||||||
| import net.minecraft.client.renderer.Tessellator; | import net.minecraft.client.renderer.Tessellator; | ||||||
| import net.minecraft.client.renderer.vertex.DefaultVertexFormats; | import net.minecraft.client.renderer.vertex.DefaultVertexFormats; | ||||||
| import net.minecraft.item.ItemStack; | import net.minecraft.item.ItemStack; | ||||||
| @@ -32,7 +32,7 @@ import static dan200.computercraft.client.gui.FixedWidthFontRenderer.*; | |||||||
| import static dan200.computercraft.client.gui.GuiComputer.*; | import static dan200.computercraft.client.gui.GuiComputer.*; | ||||||
|  |  | ||||||
| /** | /** | ||||||
|  * Emulates map rendering for pocket computers |  * Emulates map rendering for pocket computers. | ||||||
|  */ |  */ | ||||||
| @Mod.EventBusSubscriber( modid = ComputerCraft.MOD_ID, value = Dist.CLIENT ) | @Mod.EventBusSubscriber( modid = ComputerCraft.MOD_ID, value = Dist.CLIENT ) | ||||||
| public final class ItemPocketRenderer extends ItemMapLikeRenderer | public final class ItemPocketRenderer extends ItemMapLikeRenderer | ||||||
| @@ -173,7 +173,7 @@ public final class ItemPocketRenderer extends ItemMapLikeRenderer | |||||||
|     private static void renderLight( int colour, int width, int height ) |     private static void renderLight( int colour, int width, int height ) | ||||||
|     { |     { | ||||||
|         GlStateManager.enableBlend(); |         GlStateManager.enableBlend(); | ||||||
|         GlStateManager.disableTexture2D(); |         GlStateManager.disableTexture(); | ||||||
|  |  | ||||||
|         float r = ((colour >>> 16) & 0xFF) / 255.0f; |         float r = ((colour >>> 16) & 0xFF) / 255.0f; | ||||||
|         float g = ((colour >>> 8) & 0xFF) / 255.0f; |         float g = ((colour >>> 8) & 0xFF) / 255.0f; | ||||||
| @@ -188,7 +188,7 @@ public final class ItemPocketRenderer extends ItemMapLikeRenderer | |||||||
|         buffer.pos( width - LIGHT_HEIGHT * 2, height + FRAME / 2.0f, 0.0D ).color( r, g, b, 1.0f ).endVertex(); |         buffer.pos( width - LIGHT_HEIGHT * 2, height + FRAME / 2.0f, 0.0D ).color( r, g, b, 1.0f ).endVertex(); | ||||||
|  |  | ||||||
|         tessellator.draw(); |         tessellator.draw(); | ||||||
|         GlStateManager.enableTexture2D(); |         GlStateManager.enableTexture(); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     private static void renderTerminal( Terminal terminal, boolean greyscale, int width, int height ) |     private static void renderTerminal( Terminal terminal, boolean greyscale, int width, int height ) | ||||||
|   | |||||||
| @@ -6,9 +6,9 @@ | |||||||
|  |  | ||||||
| package dan200.computercraft.client.render; | package dan200.computercraft.client.render; | ||||||
|  |  | ||||||
|  | import com.mojang.blaze3d.platform.GlStateManager; | ||||||
| import dan200.computercraft.ComputerCraft; | 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.item.ItemStack; | import net.minecraft.item.ItemStack; | ||||||
| import net.minecraftforge.api.distmarker.Dist; | import net.minecraftforge.api.distmarker.Dist; | ||||||
| import net.minecraftforge.client.event.RenderItemInFrameEvent; | import net.minecraftforge.client.event.RenderItemInFrameEvent; | ||||||
| @@ -23,7 +23,7 @@ import static dan200.computercraft.shared.media.items.ItemPrintout.LINES_PER_PAG | |||||||
| import static dan200.computercraft.shared.media.items.ItemPrintout.LINE_MAX_LENGTH; | import static dan200.computercraft.shared.media.items.ItemPrintout.LINE_MAX_LENGTH; | ||||||
|  |  | ||||||
| /** | /** | ||||||
|  * Emulates map and item-frame rendering for printouts |  * Emulates map and item-frame rendering for printouts. | ||||||
|  */ |  */ | ||||||
| @Mod.EventBusSubscriber( modid = ComputerCraft.MOD_ID, value = Dist.CLIENT ) | @Mod.EventBusSubscriber( modid = ComputerCraft.MOD_ID, value = Dist.CLIENT ) | ||||||
| public final class ItemPrintoutRenderer extends ItemMapLikeRenderer | public final class ItemPrintoutRenderer extends ItemMapLikeRenderer | ||||||
|   | |||||||
| @@ -9,7 +9,7 @@ package dan200.computercraft.client.render; | |||||||
| import net.minecraft.client.renderer.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.Direction; | ||||||
| import net.minecraftforge.client.model.pipeline.IVertexConsumer; | import net.minecraftforge.client.model.pipeline.IVertexConsumer; | ||||||
| import net.minecraftforge.client.model.pipeline.LightUtil; | import net.minecraftforge.client.model.pipeline.LightUtil; | ||||||
| import net.minecraftforge.client.model.pipeline.VertexTransformer; | import net.minecraftforge.client.model.pipeline.VertexTransformer; | ||||||
| @@ -94,7 +94,7 @@ public final class ModelTransformer | |||||||
|         private final Point3f[] before = new Point3f[4]; |         private final Point3f[] before = new Point3f[4]; | ||||||
|         private final Point3f[] after = new Point3f[4]; |         private final Point3f[] after = new Point3f[4]; | ||||||
|  |  | ||||||
|         public NormalAwareTransformer( IVertexConsumer parent, Matrix4f positionMatrix, Matrix4f normalMatrix ) |         NormalAwareTransformer( IVertexConsumer parent, Matrix4f positionMatrix, Matrix4f normalMatrix ) | ||||||
|         { |         { | ||||||
|             super( parent ); |             super( parent ); | ||||||
|             this.positionMatrix = positionMatrix; |             this.positionMatrix = positionMatrix; | ||||||
| @@ -102,7 +102,7 @@ public final class ModelTransformer | |||||||
|         } |         } | ||||||
|  |  | ||||||
|         @Override |         @Override | ||||||
|         public void setQuadOrientation( @Nonnull EnumFacing orientation ) |         public void setQuadOrientation( @Nonnull Direction orientation ) | ||||||
|         { |         { | ||||||
|             super.setQuadOrientation( orientation == null ? orientation : TRSRTransformation.rotate( positionMatrix, orientation ) ); |             super.setQuadOrientation( orientation == null ? orientation : TRSRTransformation.rotate( positionMatrix, orientation ) ); | ||||||
|         } |         } | ||||||
| @@ -187,7 +187,7 @@ public final class ModelTransformer | |||||||
|         private final int[] vertexData; |         private final int[] vertexData; | ||||||
|         private int vertexIndex = 0, elementIndex = 0; |         private int vertexIndex = 0, elementIndex = 0; | ||||||
|  |  | ||||||
|         private EnumFacing orientation; |         private Direction orientation; | ||||||
|         private int quadTint; |         private int quadTint; | ||||||
|         private boolean diffuse; |         private boolean diffuse; | ||||||
|         private TextureAtlasSprite texture; |         private TextureAtlasSprite texture; | ||||||
| @@ -212,7 +212,7 @@ public final class ModelTransformer | |||||||
|         } |         } | ||||||
|  |  | ||||||
|         @Override |         @Override | ||||||
|         public void setQuadOrientation( @Nonnull EnumFacing orientation ) |         public void setQuadOrientation( @Nonnull Direction orientation ) | ||||||
|         { |         { | ||||||
|             this.orientation = orientation; |             this.orientation = orientation; | ||||||
|         } |         } | ||||||
|   | |||||||
| @@ -6,18 +6,19 @@ | |||||||
|  |  | ||||||
| package dan200.computercraft.client.render; | package dan200.computercraft.client.render; | ||||||
|  |  | ||||||
|  | import com.mojang.blaze3d.platform.GlStateManager; | ||||||
| import dan200.computercraft.ComputerCraft; | import dan200.computercraft.ComputerCraft; | ||||||
| import dan200.computercraft.shared.peripheral.monitor.TileMonitor; | import dan200.computercraft.shared.peripheral.monitor.TileMonitor; | ||||||
| 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.Tessellator; | import net.minecraft.client.renderer.Tessellator; | ||||||
| import net.minecraft.client.renderer.vertex.DefaultVertexFormats; | import net.minecraft.client.renderer.vertex.DefaultVertexFormats; | ||||||
| import net.minecraft.entity.player.EntityPlayer; |  | ||||||
| import net.minecraft.tileentity.TileEntity; | import net.minecraft.tileentity.TileEntity; | ||||||
| import net.minecraft.util.EnumFacing; | import net.minecraft.util.Direction; | ||||||
| import net.minecraft.util.math.BlockPos; | import net.minecraft.util.math.BlockPos; | ||||||
|  | import net.minecraft.util.math.BlockRayTraceResult; | ||||||
| import net.minecraft.util.math.RayTraceResult; | import net.minecraft.util.math.RayTraceResult; | ||||||
|  | import net.minecraft.util.math.Vec3d; | ||||||
| import net.minecraft.world.World; | import net.minecraft.world.World; | ||||||
| import net.minecraftforge.api.distmarker.Dist; | import net.minecraftforge.api.distmarker.Dist; | ||||||
| import net.minecraftforge.client.event.DrawBlockHighlightEvent; | import net.minecraftforge.client.event.DrawBlockHighlightEvent; | ||||||
| @@ -27,7 +28,7 @@ import org.lwjgl.opengl.GL11; | |||||||
|  |  | ||||||
| import java.util.EnumSet; | import java.util.EnumSet; | ||||||
|  |  | ||||||
| import static net.minecraft.util.EnumFacing.*; | import static net.minecraft.util.Direction.*; | ||||||
|  |  | ||||||
| @Mod.EventBusSubscriber( modid = ComputerCraft.MOD_ID, value = Dist.CLIENT ) | @Mod.EventBusSubscriber( modid = ComputerCraft.MOD_ID, value = Dist.CLIENT ) | ||||||
| public final class MonitorHighlightRenderer | public final class MonitorHighlightRenderer | ||||||
| @@ -41,10 +42,13 @@ public final class MonitorHighlightRenderer | |||||||
|     @SubscribeEvent |     @SubscribeEvent | ||||||
|     public static void drawHighlight( DrawBlockHighlightEvent event ) |     public static void drawHighlight( DrawBlockHighlightEvent event ) | ||||||
|     { |     { | ||||||
|         if( event.getTarget().type != RayTraceResult.Type.BLOCK || event.getPlayer().isSneaking() ) return; |         if( event.getTarget().getType() != RayTraceResult.Type.BLOCK || event.getInfo().getRenderViewEntity().isSneaking() ) | ||||||
|  |         { | ||||||
|  |             return; | ||||||
|  |         } | ||||||
|  |  | ||||||
|         World world = event.getPlayer().getEntityWorld(); |         World world = event.getInfo().getRenderViewEntity().getEntityWorld(); | ||||||
|         BlockPos pos = event.getTarget().getBlockPos(); |         BlockPos pos = ((BlockRayTraceResult) event.getTarget()).getPos(); | ||||||
|  |  | ||||||
|         TileEntity tile = world.getTileEntity( pos ); |         TileEntity tile = world.getTileEntity( pos ); | ||||||
|         if( !(tile instanceof TileMonitor) ) return; |         if( !(tile instanceof TileMonitor) ) return; | ||||||
| @@ -53,8 +57,8 @@ public final class MonitorHighlightRenderer | |||||||
|         event.setCanceled( true ); |         event.setCanceled( true ); | ||||||
|  |  | ||||||
|         // Determine which sides are part of the external faces of the monitor, and so which need to be rendered. |         // Determine which sides are part of the external faces of the monitor, and so which need to be rendered. | ||||||
|         EnumSet<EnumFacing> faces = EnumSet.allOf( EnumFacing.class ); |         EnumSet<Direction> faces = EnumSet.allOf( Direction.class ); | ||||||
|         EnumFacing front = monitor.getFront(); |         Direction front = monitor.getFront(); | ||||||
|         faces.remove( front ); |         faces.remove( front ); | ||||||
|         if( monitor.getXIndex() != 0 ) faces.remove( monitor.getRight().getOpposite() ); |         if( monitor.getXIndex() != 0 ) faces.remove( monitor.getRight().getOpposite() ); | ||||||
|         if( monitor.getXIndex() != monitor.getWidth() - 1 ) faces.remove( monitor.getRight() ); |         if( monitor.getXIndex() != monitor.getWidth() - 1 ) faces.remove( monitor.getRight() ); | ||||||
| @@ -62,18 +66,14 @@ public final class MonitorHighlightRenderer | |||||||
|         if( monitor.getYIndex() != monitor.getHeight() - 1 ) faces.remove( monitor.getDown() ); |         if( monitor.getYIndex() != monitor.getHeight() - 1 ) faces.remove( monitor.getDown() ); | ||||||
|  |  | ||||||
|         GlStateManager.enableBlend(); |         GlStateManager.enableBlend(); | ||||||
|         GlStateManager.blendFuncSeparate(GlStateManager.SourceFactor.SRC_ALPHA, GlStateManager.DestFactor.ONE_MINUS_SRC_ALPHA, GlStateManager.SourceFactor.ONE, GlStateManager.DestFactor.ZERO); |         GlStateManager.blendFuncSeparate( GlStateManager.SourceFactor.SRC_ALPHA, GlStateManager.DestFactor.ONE_MINUS_SRC_ALPHA, GlStateManager.SourceFactor.ONE, GlStateManager.DestFactor.ZERO ); | ||||||
|         GlStateManager.lineWidth(Math.max(2.5F, (float) Minecraft.getInstance().mainWindow.getFramebufferWidth() / 1920.0F * 2.5F)); |         GlStateManager.lineWidth( Math.max( 2.5F, (float) Minecraft.getInstance().mainWindow.getFramebufferWidth() / 1920.0F * 2.5F ) ); | ||||||
|         GlStateManager.disableTexture2D(); |         GlStateManager.disableTexture(); | ||||||
|         GlStateManager.depthMask( false ); |         GlStateManager.depthMask( false ); | ||||||
|         GlStateManager.pushMatrix(); |         GlStateManager.pushMatrix(); | ||||||
|  |  | ||||||
|         EntityPlayer player = event.getPlayer(); |         Vec3d cameraPos = event.getInfo().getProjectedView(); | ||||||
|         double x = player.lastTickPosX + (player.posX - player.lastTickPosX) * event.getPartialTicks(); |         GlStateManager.translated( pos.getX() - cameraPos.getX(), pos.getY() - cameraPos.getY(), pos.getZ() - cameraPos.getZ() ); | ||||||
|         double y = player.lastTickPosY + (player.posY - player.lastTickPosY) * event.getPartialTicks(); |  | ||||||
|         double z = player.lastTickPosZ + (player.posZ - player.lastTickPosZ) * event.getPartialTicks(); |  | ||||||
|  |  | ||||||
|         GlStateManager.translated( -x + pos.getX(), -y + pos.getY(), -z + pos.getZ() ); |  | ||||||
|  |  | ||||||
|         Tessellator tessellator = Tessellator.getInstance(); |         Tessellator tessellator = Tessellator.getInstance(); | ||||||
|         BufferBuilder buffer = tessellator.getBuffer(); |         BufferBuilder buffer = tessellator.getBuffer(); | ||||||
| @@ -97,11 +97,11 @@ public final class MonitorHighlightRenderer | |||||||
|  |  | ||||||
|         GlStateManager.popMatrix(); |         GlStateManager.popMatrix(); | ||||||
|         GlStateManager.depthMask( true ); |         GlStateManager.depthMask( true ); | ||||||
|         GlStateManager.enableTexture2D(); |         GlStateManager.enableTexture(); | ||||||
|         GlStateManager.disableBlend(); |         GlStateManager.disableBlend(); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     private static void line( BufferBuilder buffer, int x, int y, int z, EnumFacing direction ) |     private static void line( BufferBuilder buffer, int x, int y, int z, Direction direction ) | ||||||
|     { |     { | ||||||
|         double minX = x == 0 ? -EXPAND : 1 + EXPAND; |         double minX = x == 0 ? -EXPAND : 1 + EXPAND; | ||||||
|         double minY = y == 0 ? -EXPAND : 1 + EXPAND; |         double minY = y == 0 ? -EXPAND : 1 + EXPAND; | ||||||
|   | |||||||
| @@ -6,14 +6,14 @@ | |||||||
|  |  | ||||||
| package dan200.computercraft.client.render; | package dan200.computercraft.client.render; | ||||||
|  |  | ||||||
|  | import com.mojang.blaze3d.platform.GlStateManager; | ||||||
|  | import com.mojang.blaze3d.platform.GlStateManager.DestFactor; | ||||||
|  | import com.mojang.blaze3d.platform.GlStateManager.SourceFactor; | ||||||
| import dan200.computercraft.client.gui.FixedWidthFontRenderer; | import dan200.computercraft.client.gui.FixedWidthFontRenderer; | ||||||
| import dan200.computercraft.core.terminal.TextBuffer; | import dan200.computercraft.core.terminal.TextBuffer; | ||||||
| import dan200.computercraft.shared.util.Palette; | import dan200.computercraft.shared.util.Palette; | ||||||
| import net.minecraft.client.Minecraft; | import net.minecraft.client.Minecraft; | ||||||
| import net.minecraft.client.renderer.BufferBuilder; | import net.minecraft.client.renderer.BufferBuilder; | ||||||
| import net.minecraft.client.renderer.GlStateManager; |  | ||||||
| import net.minecraft.client.renderer.GlStateManager.DestFactor; |  | ||||||
| import net.minecraft.client.renderer.GlStateManager.SourceFactor; |  | ||||||
| import net.minecraft.client.renderer.Tessellator; | import net.minecraft.client.renderer.Tessellator; | ||||||
| import net.minecraft.client.renderer.vertex.DefaultVertexFormats; | import net.minecraft.client.renderer.vertex.DefaultVertexFormats; | ||||||
| import net.minecraft.util.ResourceLocation; | import net.minecraft.util.ResourceLocation; | ||||||
| @@ -28,32 +28,32 @@ public final class PrintoutRenderer | |||||||
|     private static final double BG_SIZE = 256.0; |     private static final double BG_SIZE = 256.0; | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * Width of a page |      * Width of a page. | ||||||
|      */ |      */ | ||||||
|     public static final int X_SIZE = 172; |     public static final int X_SIZE = 172; | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * Height of a page |      * Height of a page. | ||||||
|      */ |      */ | ||||||
|     public static final int Y_SIZE = 209; |     public static final int Y_SIZE = 209; | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * Padding between the left and right of a page and the text |      * Padding between the left and right of a page and the text. | ||||||
|      */ |      */ | ||||||
|     public static final int X_TEXT_MARGIN = 13; |     public static final int X_TEXT_MARGIN = 13; | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * Padding between the top and bottom of a page and the text |      * Padding between the top and bottom of a page and the text. | ||||||
|      */ |      */ | ||||||
|     public static final int Y_TEXT_MARGIN = 11; |     public static final int Y_TEXT_MARGIN = 11; | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * Width of the extra page texture |      * Width of the extra page texture. | ||||||
|      */ |      */ | ||||||
|     private static final int X_FOLD_SIZE = 12; |     private static final int X_FOLD_SIZE = 12; | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * Size of the leather cover |      * Size of the leather cover. | ||||||
|      */ |      */ | ||||||
|     public static final int COVER_SIZE = 12; |     public static final int COVER_SIZE = 12; | ||||||
|  |  | ||||||
| @@ -76,7 +76,7 @@ public final class PrintoutRenderer | |||||||
|     { |     { | ||||||
|         GlStateManager.color4f( 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.enableTexture(); | ||||||
|         GlStateManager.blendFuncSeparate( 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(); | ||||||
| @@ -91,7 +91,7 @@ public final class PrintoutRenderer | |||||||
|     { |     { | ||||||
|         GlStateManager.color4f( 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.enableTexture(); | ||||||
|         GlStateManager.blendFuncSeparate( 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.getInstance().getTextureManager().bindTexture( BG ); |         Minecraft.getInstance().getTextureManager().bindTexture( BG ); | ||||||
|   | |||||||
| @@ -6,6 +6,7 @@ | |||||||
|  |  | ||||||
| package dan200.computercraft.client.render; | package dan200.computercraft.client.render; | ||||||
|  |  | ||||||
|  | import com.mojang.blaze3d.platform.GlStateManager; | ||||||
| import dan200.computercraft.ComputerCraft; | import dan200.computercraft.ComputerCraft; | ||||||
| import dan200.computercraft.shared.peripheral.modem.wired.BlockCable; | import dan200.computercraft.shared.peripheral.modem.wired.BlockCable; | ||||||
| import dan200.computercraft.shared.peripheral.modem.wired.CableModemVariant; | import dan200.computercraft.shared.peripheral.modem.wired.CableModemVariant; | ||||||
| @@ -13,10 +14,9 @@ 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 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.BlockState; | ||||||
| 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.Tessellator; | import net.minecraft.client.renderer.Tessellator; | ||||||
| import net.minecraft.client.renderer.WorldRenderer; | import net.minecraft.client.renderer.WorldRenderer; | ||||||
| import net.minecraft.client.renderer.model.IBakedModel; | import net.minecraft.client.renderer.model.IBakedModel; | ||||||
| @@ -24,11 +24,12 @@ import net.minecraft.client.renderer.texture.TextureAtlasSprite; | |||||||
| import net.minecraft.client.renderer.tileentity.TileEntityRenderer; | 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.ResourceLocation; | ||||||
| import net.minecraft.util.math.BlockPos; | import net.minecraft.util.math.BlockPos; | ||||||
|  | import net.minecraft.util.math.BlockRayTraceResult; | ||||||
| import net.minecraft.util.math.RayTraceResult; | import net.minecraft.util.math.RayTraceResult; | ||||||
| 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.model.data.EmptyModelData; | import net.minecraftforge.client.model.data.EmptyModelData; | ||||||
| import org.lwjgl.opengl.GL11; | import org.lwjgl.opengl.GL11; | ||||||
|  |  | ||||||
| @@ -40,8 +41,17 @@ import java.util.Random; | |||||||
|  */ |  */ | ||||||
| public class TileEntityCableRenderer extends TileEntityRenderer<TileCable> | public class TileEntityCableRenderer extends TileEntityRenderer<TileCable> | ||||||
| { | { | ||||||
|  |     private static final ResourceLocation[] DESTROY_STAGES = new ResourceLocation[10]; | ||||||
|     private static final Random random = new Random(); |     private static final Random random = new Random(); | ||||||
|  |  | ||||||
|  |     static | ||||||
|  |     { | ||||||
|  |         for( int i = 0; i < DESTROY_STAGES.length; i++ ) | ||||||
|  |         { | ||||||
|  |             DESTROY_STAGES[i] = new ResourceLocation( "block/destroy_stage_" + i ); | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|     @Override |     @Override | ||||||
|     public void render( @Nonnull TileCable te, double x, double y, double z, float partialTicks, int destroyStage ) |     public void render( @Nonnull TileCable te, double x, double y, double z, float partialTicks, int destroyStage ) | ||||||
|     { |     { | ||||||
| @@ -52,16 +62,17 @@ public class TileEntityCableRenderer extends TileEntityRenderer<TileCable> | |||||||
|         Minecraft mc = Minecraft.getInstance(); |         Minecraft mc = Minecraft.getInstance(); | ||||||
|  |  | ||||||
|         RayTraceResult hit = mc.objectMouseOver; |         RayTraceResult hit = mc.objectMouseOver; | ||||||
|         if( hit == null || !hit.getBlockPos().equals( pos ) ) return; |         if( hit == null || hit.getType() != RayTraceResult.Type.BLOCK || !((BlockRayTraceResult) hit).getPos().equals( pos ) ) | ||||||
|  |         { | ||||||
|         if( MinecraftForgeClient.getRenderPass() != 0 ) return; |             return; | ||||||
|  |         } | ||||||
|  |  | ||||||
|         World world = te.getWorld(); |         World world = te.getWorld(); | ||||||
|         IBlockState state = world.getBlockState( pos ); |         BlockState state = world.getBlockState( pos ); | ||||||
|         Block block = state.getBlock(); |         Block block = state.getBlock(); | ||||||
|         if( block != ComputerCraft.Blocks.cable ) return; |         if( block != ComputerCraft.Blocks.cable ) return; | ||||||
|  |  | ||||||
|         state = WorldUtil.isVecInside( CableShapes.getModemShape( state ), hit.hitVec.subtract( pos.getX(), pos.getY(), pos.getZ() ) ) |         state = WorldUtil.isVecInside( CableShapes.getModemShape( state ), hit.getHitVec().subtract( pos.getX(), pos.getY(), pos.getZ() ) ) | ||||||
|             ? block.getDefaultState().with( BlockCable.MODEM, state.get( BlockCable.MODEM ) ) |             ? block.getDefaultState().with( BlockCable.MODEM, state.get( BlockCable.MODEM ) ) | ||||||
|             : state.with( BlockCable.MODEM, CableModemVariant.None ); |             : state.with( BlockCable.MODEM, CableModemVariant.None ); | ||||||
|  |  | ||||||
| @@ -69,18 +80,19 @@ public class TileEntityCableRenderer extends TileEntityRenderer<TileCable> | |||||||
|  |  | ||||||
|         preRenderDamagedBlocks(); |         preRenderDamagedBlocks(); | ||||||
|  |  | ||||||
|  |         ForgeHooksClient.setRenderLayer( block.getRenderLayer() ); | ||||||
|  |  | ||||||
|  |         // See BlockRendererDispatcher#renderBlockDamage | ||||||
|  |         TextureAtlasSprite breakingTexture = mc.getTextureMap().getSprite( DESTROY_STAGES[destroyStage] ); | ||||||
|  |  | ||||||
|         BufferBuilder buffer = Tessellator.getInstance().getBuffer(); |         BufferBuilder buffer = Tessellator.getInstance().getBuffer(); | ||||||
|         buffer.begin( GL11.GL_QUADS, DefaultVertexFormats.BLOCK ); |         buffer.begin( GL11.GL_QUADS, DefaultVertexFormats.BLOCK ); | ||||||
|         buffer.setTranslation( x - pos.getX(), y - pos.getY(), z - pos.getZ() ); |         buffer.setTranslation( x - pos.getX(), y - pos.getY(), z - pos.getZ() ); | ||||||
|         buffer.noColor(); |         buffer.noColor(); | ||||||
|  |  | ||||||
|         ForgeHooksClient.setRenderLayer( block.getRenderLayer() ); |  | ||||||
|  |  | ||||||
|         // See BlockRendererDispatcher#renderBlockDamage |  | ||||||
|         TextureAtlasSprite breakingTexture = mc.getTextureMap().getSprite( DESTROY_STAGES[destroyStage] ); |  | ||||||
|         mc.getBlockRendererDispatcher().getBlockModelRenderer().renderModel( |         mc.getBlockRendererDispatcher().getBlockModelRenderer().renderModel( | ||||||
|             world, |             world, | ||||||
|             ForgeHooksClient.getDamageModel( model, breakingTexture, state, world, pos ), |             ForgeHooksClient.getDamageModel( model, breakingTexture, state, world, pos, 0 ), | ||||||
|             state, pos, buffer, true, random, state.getPositionRandom( pos ), EmptyModelData.INSTANCE |             state, pos, buffer, true, random, state.getPositionRandom( pos ), EmptyModelData.INSTANCE | ||||||
|         ); |         ); | ||||||
|  |  | ||||||
| @@ -93,6 +105,8 @@ public class TileEntityCableRenderer extends TileEntityRenderer<TileCable> | |||||||
|     } |     } | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|  |      * Set up the state for rendering block-breaking progress. | ||||||
|  |      * | ||||||
|      * @see WorldRenderer#preRenderDamagedBlocks() |      * @see WorldRenderer#preRenderDamagedBlocks() | ||||||
|      */ |      */ | ||||||
|     private void preRenderDamagedBlocks() |     private void preRenderDamagedBlocks() | ||||||
| @@ -111,6 +125,8 @@ public class TileEntityCableRenderer extends TileEntityRenderer<TileCable> | |||||||
|     } |     } | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|  |      * Tear down the state for rendering block-breaking progress. | ||||||
|  |      * | ||||||
|      * @see WorldRenderer#postRenderDamagedBlocks() |      * @see WorldRenderer#postRenderDamagedBlocks() | ||||||
|      */ |      */ | ||||||
|     private void postRenderDamagedBlocks() |     private void postRenderDamagedBlocks() | ||||||
|   | |||||||
| @@ -6,6 +6,8 @@ | |||||||
|  |  | ||||||
| package dan200.computercraft.client.render; | package dan200.computercraft.client.render; | ||||||
|  |  | ||||||
|  | import com.mojang.blaze3d.platform.GLX; | ||||||
|  | import com.mojang.blaze3d.platform.GlStateManager; | ||||||
| import dan200.computercraft.client.FrameInfo; | import dan200.computercraft.client.FrameInfo; | ||||||
| import dan200.computercraft.client.gui.FixedWidthFontRenderer; | import dan200.computercraft.client.gui.FixedWidthFontRenderer; | ||||||
| import dan200.computercraft.core.terminal.Terminal; | import dan200.computercraft.core.terminal.Terminal; | ||||||
| @@ -17,12 +19,10 @@ import dan200.computercraft.shared.util.DirectionUtil; | |||||||
| import dan200.computercraft.shared.util.Palette; | import dan200.computercraft.shared.util.Palette; | ||||||
| import net.minecraft.client.Minecraft; | import net.minecraft.client.Minecraft; | ||||||
| import net.minecraft.client.renderer.BufferBuilder; | import net.minecraft.client.renderer.BufferBuilder; | ||||||
| import net.minecraft.client.renderer.GlStateManager; |  | ||||||
| import net.minecraft.client.renderer.OpenGlHelper; |  | ||||||
| import net.minecraft.client.renderer.Tessellator; | import net.minecraft.client.renderer.Tessellator; | ||||||
| import net.minecraft.client.renderer.tileentity.TileEntityRenderer; | 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.Direction; | ||||||
| import net.minecraft.util.math.BlockPos; | import net.minecraft.util.math.BlockPos; | ||||||
| import org.lwjgl.opengl.GL11; | import org.lwjgl.opengl.GL11; | ||||||
|  |  | ||||||
| @@ -64,8 +64,8 @@ public class TileEntityMonitorRenderer extends TileEntityRenderer<TileMonitor> | |||||||
|         posZ += originPos.getZ() - monitorPos.getZ(); |         posZ += originPos.getZ() - monitorPos.getZ(); | ||||||
|  |  | ||||||
|         // Determine orientation |         // Determine orientation | ||||||
|         EnumFacing dir = origin.getDirection(); |         Direction dir = origin.getDirection(); | ||||||
|         EnumFacing front = origin.getFront(); |         Direction front = origin.getFront(); | ||||||
|         float yaw = dir.getHorizontalAngle(); |         float yaw = dir.getHorizontalAngle(); | ||||||
|         float pitch = DirectionUtil.toPitchAngle( front ); |         float pitch = DirectionUtil.toPitchAngle( front ); | ||||||
|  |  | ||||||
| @@ -94,7 +94,7 @@ public class TileEntityMonitorRenderer extends TileEntityRenderer<TileMonitor> | |||||||
|  |  | ||||||
|             // Draw the contents |             // Draw the contents | ||||||
|             GlStateManager.depthMask( false ); |             GlStateManager.depthMask( false ); | ||||||
|             OpenGlHelper.glMultiTexCoord2f( OpenGlHelper.GL_TEXTURE1, 0xFFFF, 0xFFFF ); |             GLX.glMultiTexCoord2f( GLX.GL_TEXTURE1, 0xFFFF, 0xFFFF ); | ||||||
|             GlStateManager.disableLighting(); |             GlStateManager.disableLighting(); | ||||||
|             mc.gameRenderer.disableLightmap(); |             mc.gameRenderer.disableLightmap(); | ||||||
|             try |             try | ||||||
| @@ -171,7 +171,7 @@ public class TileEntityMonitorRenderer extends TileEntityRenderer<TileMonitor> | |||||||
|                             } |                             } | ||||||
|                         } |                         } | ||||||
|                         GlStateManager.callList( originTerminal.renderDisplayLists[0] ); |                         GlStateManager.callList( originTerminal.renderDisplayLists[0] ); | ||||||
|                         GlStateManager.resetColor(); |                         GlStateManager.clearCurrentColor(); | ||||||
|  |  | ||||||
|                         // Draw text |                         // Draw text | ||||||
|                         fontRenderer.bindFont(); |                         fontRenderer.bindFont(); | ||||||
| @@ -199,7 +199,7 @@ public class TileEntityMonitorRenderer extends TileEntityRenderer<TileMonitor> | |||||||
|                             } |                             } | ||||||
|                         } |                         } | ||||||
|                         GlStateManager.callList( originTerminal.renderDisplayLists[1] ); |                         GlStateManager.callList( originTerminal.renderDisplayLists[1] ); | ||||||
|                         GlStateManager.resetColor(); |                         GlStateManager.clearCurrentColor(); | ||||||
|  |  | ||||||
|                         // Draw cursor |                         // Draw cursor | ||||||
|                         fontRenderer.bindFont(); |                         fontRenderer.bindFont(); | ||||||
| @@ -233,7 +233,7 @@ public class TileEntityMonitorRenderer extends TileEntityRenderer<TileMonitor> | |||||||
|                         if( FrameInfo.getGlobalCursorBlink() ) |                         if( FrameInfo.getGlobalCursorBlink() ) | ||||||
|                         { |                         { | ||||||
|                             GlStateManager.callList( originTerminal.renderDisplayLists[2] ); |                             GlStateManager.callList( originTerminal.renderDisplayLists[2] ); | ||||||
|                             GlStateManager.resetColor(); |                             GlStateManager.clearCurrentColor(); | ||||||
|                         } |                         } | ||||||
|                     } |                     } | ||||||
|                     finally |                     finally | ||||||
|   | |||||||
| @@ -6,6 +6,7 @@ | |||||||
|  |  | ||||||
| package dan200.computercraft.client.render; | package dan200.computercraft.client.render; | ||||||
|  |  | ||||||
|  | import com.mojang.blaze3d.platform.GlStateManager; | ||||||
| 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.computer.core.ComputerFamily; | import dan200.computercraft.shared.computer.core.ComputerFamily; | ||||||
| @@ -13,22 +14,23 @@ import dan200.computercraft.shared.turtle.blocks.TileTurtle; | |||||||
| import dan200.computercraft.shared.util.DirectionUtil; | 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.BlockState; | ||||||
| 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.GameRenderer; | import net.minecraft.client.renderer.GameRenderer; | ||||||
| import net.minecraft.client.renderer.GlStateManager; |  | ||||||
| import net.minecraft.client.renderer.Tessellator; | import net.minecraft.client.renderer.Tessellator; | ||||||
| import net.minecraft.client.renderer.model.BakedQuad; | import net.minecraft.client.renderer.model.BakedQuad; | ||||||
| import net.minecraft.client.renderer.model.IBakedModel; | import net.minecraft.client.renderer.model.IBakedModel; | ||||||
| import net.minecraft.client.renderer.model.ModelManager; | import net.minecraft.client.renderer.model.ModelManager; | ||||||
| import net.minecraft.client.renderer.model.ModelResourceLocation; | import net.minecraft.client.renderer.model.ModelResourceLocation; | ||||||
| import net.minecraft.client.renderer.texture.TextureMap; | import net.minecraft.client.renderer.texture.AtlasTexture; | ||||||
| import net.minecraft.client.renderer.tileentity.TileEntityRenderer; | 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.Direction; | ||||||
| import net.minecraft.util.ResourceLocation; | import net.minecraft.util.ResourceLocation; | ||||||
|  | import net.minecraft.util.math.BlockRayTraceResult; | ||||||
|  | import net.minecraft.util.math.RayTraceResult; | ||||||
| import net.minecraft.util.math.Vec3d; | import net.minecraft.util.math.Vec3d; | ||||||
| import net.minecraftforge.client.ForgeHooksClient; | import net.minecraftforge.client.ForgeHooksClient; | ||||||
| import net.minecraftforge.client.model.data.EmptyModelData; | import net.minecraftforge.client.model.data.EmptyModelData; | ||||||
| @@ -85,13 +87,14 @@ public class TileEntityTurtleRenderer extends TileEntityRenderer<TileTurtle> | |||||||
|     { |     { | ||||||
|         // Render the label |         // Render the label | ||||||
|         String label = turtle.createProxy().getLabel(); |         String label = turtle.createProxy().getLabel(); | ||||||
|         if( label != null && rendererDispatcher.cameraHitResult != null && turtle.getPos().equals( rendererDispatcher.cameraHitResult.getBlockPos() ) ) |         RayTraceResult hit = rendererDispatcher.cameraHitResult; | ||||||
|  |         if( label != null && hit.getType() == RayTraceResult.Type.BLOCK && turtle.getPos().equals( ((BlockRayTraceResult) hit).getPos() ) ) | ||||||
|         { |         { | ||||||
|             setLightmapDisabled( true ); |             setLightmapDisabled( true ); | ||||||
|             GameRenderer.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.renderInfo.getYaw(), rendererDispatcher.renderInfo.getPitch(), false | ||||||
|             ); |             ); | ||||||
|             setLightmapDisabled( false ); |             setLightmapDisabled( false ); | ||||||
|         } |         } | ||||||
| @@ -99,7 +102,7 @@ public class TileEntityTurtleRenderer extends TileEntityRenderer<TileTurtle> | |||||||
|         GlStateManager.pushMatrix(); |         GlStateManager.pushMatrix(); | ||||||
|         try |         try | ||||||
|         { |         { | ||||||
|             IBlockState state = turtle.getBlockState(); |             BlockState 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 ); | ||||||
| @@ -153,7 +156,7 @@ public class TileEntityTurtleRenderer extends TileEntityRenderer<TileTurtle> | |||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     private void renderUpgrade( IBlockState state, TileTurtle turtle, TurtleSide side, float f ) |     private void renderUpgrade( BlockState state, TileTurtle turtle, TurtleSide side, float f ) | ||||||
|     { |     { | ||||||
|         ITurtleUpgrade upgrade = turtle.getUpgrade( side ); |         ITurtleUpgrade upgrade = turtle.getUpgrade( side ); | ||||||
|         if( upgrade != null ) |         if( upgrade != null ) | ||||||
| @@ -186,20 +189,20 @@ public class TileEntityTurtleRenderer extends TileEntityRenderer<TileTurtle> | |||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     private void renderModel( IBlockState state, ModelResourceLocation modelLocation, int[] tints ) |     private void renderModel( BlockState state, ModelResourceLocation modelLocation, int[] tints ) | ||||||
|     { |     { | ||||||
|         Minecraft mc = Minecraft.getInstance(); |         Minecraft mc = Minecraft.getInstance(); | ||||||
|         ModelManager modelManager = mc.getItemRenderer().getItemModelMesher().getModelManager(); |         ModelManager modelManager = mc.getItemRenderer().getItemModelMesher().getModelManager(); | ||||||
|         renderModel( state, modelManager.getModel( modelLocation ), tints ); |         renderModel( state, modelManager.getModel( modelLocation ), tints ); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     private void renderModel( IBlockState state, IBakedModel model, int[] tints ) |     private void renderModel( BlockState state, IBakedModel model, int[] tints ) | ||||||
|     { |     { | ||||||
|         Random random = new Random( 0 ); |         Random random = new Random( 0 ); | ||||||
|         Tessellator tessellator = Tessellator.getInstance(); |         Tessellator tessellator = Tessellator.getInstance(); | ||||||
|         rendererDispatcher.textureManager.bindTexture( TextureMap.LOCATION_BLOCKS_TEXTURE ); |         rendererDispatcher.textureManager.bindTexture( AtlasTexture.LOCATION_BLOCKS_TEXTURE ); | ||||||
|         renderQuads( tessellator, model.getQuads( state, null, random, EmptyModelData.INSTANCE ), tints ); |         renderQuads( tessellator, model.getQuads( state, null, random, EmptyModelData.INSTANCE ), tints ); | ||||||
|         for( EnumFacing facing : DirectionUtil.FACINGS ) |         for( Direction facing : DirectionUtil.FACINGS ) | ||||||
|         { |         { | ||||||
|             renderQuads( tessellator, model.getQuads( state, facing, random, EmptyModelData.INSTANCE ), tints ); |             renderQuads( tessellator, model.getQuads( state, facing, random, EmptyModelData.INSTANCE ), tints ); | ||||||
|         } |         } | ||||||
|   | |||||||
| @@ -9,15 +9,15 @@ package dan200.computercraft.client.render; | |||||||
| import dan200.computercraft.ComputerCraft; | import dan200.computercraft.ComputerCraft; | ||||||
| import net.minecraft.client.renderer.model.IBakedModel; | import net.minecraft.client.renderer.model.IBakedModel; | ||||||
| import net.minecraft.client.renderer.model.IUnbakedModel; | import net.minecraft.client.renderer.model.IUnbakedModel; | ||||||
|  | import net.minecraft.client.renderer.model.ModelBakery; | ||||||
|  | import net.minecraft.client.renderer.texture.ISprite; | ||||||
| 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.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.common.model.IModelState; |  | ||||||
|  |  | ||||||
| import javax.annotation.Nonnull; | import javax.annotation.Nonnull; | ||||||
| import javax.annotation.Nullable; |  | ||||||
| import java.util.Arrays; | import java.util.Arrays; | ||||||
| import java.util.Collection; | import java.util.Collection; | ||||||
| import java.util.Set; | import java.util.Set; | ||||||
| @@ -70,7 +70,10 @@ public final class TurtleModelLoader implements ICustomModelLoader | |||||||
|     { |     { | ||||||
|         private final ResourceLocation family; |         private final ResourceLocation family; | ||||||
|  |  | ||||||
|         private TurtleModel( ResourceLocation family ) {this.family = family;} |         private TurtleModel( ResourceLocation family ) | ||||||
|  |         { | ||||||
|  |             this.family = family; | ||||||
|  |         } | ||||||
|  |  | ||||||
|         @Nonnull |         @Nonnull | ||||||
|         @Override |         @Override | ||||||
| @@ -88,13 +91,13 @@ public final class TurtleModelLoader implements ICustomModelLoader | |||||||
|                 .collect( Collectors.toSet() ); |                 .collect( Collectors.toSet() ); | ||||||
|         } |         } | ||||||
|  |  | ||||||
|         @Nullable |         @Nonnull | ||||||
|         @Override |         @Override | ||||||
|         public IBakedModel bake( @Nonnull Function<ResourceLocation, IUnbakedModel> modelGetter, @Nonnull Function<ResourceLocation, TextureAtlasSprite> spriteGetter, @Nonnull IModelState state, boolean uvlock, @Nonnull VertexFormat format ) |         public IBakedModel bake( @Nonnull ModelBakery bakery, @Nonnull Function<ResourceLocation, TextureAtlasSprite> spriteGetter, @Nonnull ISprite sprite, @Nonnull VertexFormat format ) | ||||||
|         { |         { | ||||||
|             return new TurtleSmartItemModel( |             return new TurtleSmartItemModel( | ||||||
|                 modelGetter.apply( family ).bake( modelGetter, spriteGetter, state, uvlock, format ), |                 bakery.getBakedModel( family, sprite, spriteGetter, format ), | ||||||
|                 modelGetter.apply( COLOUR_TURTLE_MODEL ).bake( modelGetter, spriteGetter, state, uvlock, format ) |                 bakery.getBakedModel( COLOUR_TURTLE_MODEL, sprite, spriteGetter, format ) | ||||||
|             ); |             ); | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
|   | |||||||
| @@ -6,12 +6,12 @@ | |||||||
|  |  | ||||||
| package dan200.computercraft.client.render; | package dan200.computercraft.client.render; | ||||||
|  |  | ||||||
| import net.minecraft.block.state.IBlockState; | import net.minecraft.block.BlockState; | ||||||
| import net.minecraft.client.renderer.model.BakedQuad; | import net.minecraft.client.renderer.model.BakedQuad; | ||||||
| import net.minecraft.client.renderer.model.IBakedModel; | import net.minecraft.client.renderer.model.IBakedModel; | ||||||
| import net.minecraft.client.renderer.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.Direction; | ||||||
| import net.minecraftforge.client.model.data.EmptyModelData; | import net.minecraftforge.client.model.data.EmptyModelData; | ||||||
| import net.minecraftforge.client.model.data.IModelData; | import net.minecraftforge.client.model.data.IModelData; | ||||||
|  |  | ||||||
| @@ -29,7 +29,7 @@ public class TurtleMultiModel implements IBakedModel | |||||||
|     private final IBakedModel m_rightUpgradeModel; |     private final IBakedModel m_rightUpgradeModel; | ||||||
|     private final Matrix4f m_rightUpgradeTransform; |     private final Matrix4f m_rightUpgradeTransform; | ||||||
|     private List<BakedQuad> m_generalQuads = null; |     private List<BakedQuad> m_generalQuads = null; | ||||||
|     private Map<EnumFacing, List<BakedQuad>> m_faceQuads = new EnumMap<>( EnumFacing.class ); |     private Map<Direction, List<BakedQuad>> m_faceQuads = new EnumMap<>( Direction.class ); | ||||||
|  |  | ||||||
|     public TurtleMultiModel( IBakedModel baseModel, IBakedModel overlayModel, Matrix4f generalTransform, IBakedModel leftUpgradeModel, Matrix4f leftUpgradeTransform, IBakedModel rightUpgradeModel, Matrix4f rightUpgradeTransform ) |     public TurtleMultiModel( IBakedModel baseModel, IBakedModel overlayModel, Matrix4f generalTransform, IBakedModel leftUpgradeModel, Matrix4f leftUpgradeTransform, IBakedModel rightUpgradeModel, Matrix4f rightUpgradeTransform ) | ||||||
|     { |     { | ||||||
| @@ -46,14 +46,14 @@ public class TurtleMultiModel implements IBakedModel | |||||||
|     @Nonnull |     @Nonnull | ||||||
|     @Override |     @Override | ||||||
|     @Deprecated |     @Deprecated | ||||||
|     public List<BakedQuad> getQuads( IBlockState state, EnumFacing side, @Nonnull Random rand ) |     public List<BakedQuad> getQuads( BlockState state, Direction side, @Nonnull Random rand ) | ||||||
|     { |     { | ||||||
|         return getQuads( state, side, rand, EmptyModelData.INSTANCE ); |         return getQuads( state, side, rand, EmptyModelData.INSTANCE ); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @Nonnull |     @Nonnull | ||||||
|     @Override |     @Override | ||||||
|     public List<BakedQuad> getQuads( IBlockState state, EnumFacing side, @Nonnull Random rand, @Nonnull IModelData data ) |     public List<BakedQuad> getQuads( BlockState state, Direction side, @Nonnull Random rand, @Nonnull IModelData data ) | ||||||
|     { |     { | ||||||
|         if( side != null ) |         if( side != null ) | ||||||
|         { |         { | ||||||
| @@ -67,7 +67,7 @@ public class TurtleMultiModel implements IBakedModel | |||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     private List<BakedQuad> buildQuads( IBlockState state, EnumFacing side, Random rand ) |     private List<BakedQuad> buildQuads( BlockState state, Direction side, Random rand ) | ||||||
|     { |     { | ||||||
|         ArrayList<BakedQuad> quads = new ArrayList<>(); |         ArrayList<BakedQuad> quads = new ArrayList<>(); | ||||||
|         ModelTransformer.transformQuadsTo( quads, m_baseModel.getQuads( state, side, rand, EmptyModelData.INSTANCE ), m_generalTransform ); |         ModelTransformer.transformQuadsTo( quads, m_baseModel.getQuads( state, side, rand, EmptyModelData.INSTANCE ), m_generalTransform ); | ||||||
| @@ -93,7 +93,7 @@ public class TurtleMultiModel implements IBakedModel | |||||||
|                 upgradeTransform = new Matrix4f( m_generalTransform ); |                 upgradeTransform = new Matrix4f( m_generalTransform ); | ||||||
|                 upgradeTransform.mul( m_rightUpgradeTransform ); |                 upgradeTransform.mul( m_rightUpgradeTransform ); | ||||||
|             } |             } | ||||||
|             ModelTransformer.transformQuadsTo( quads, m_rightUpgradeModel.getQuads( state, side, rand , EmptyModelData.INSTANCE), upgradeTransform ); |             ModelTransformer.transformQuadsTo( quads, m_rightUpgradeModel.getQuads( state, side, rand, EmptyModelData.INSTANCE ), upgradeTransform ); | ||||||
|         } |         } | ||||||
|         quads.trimToSize(); |         quads.trimToSize(); | ||||||
|         return quads; |         return quads; | ||||||
| @@ -119,6 +119,7 @@ public class TurtleMultiModel implements IBakedModel | |||||||
|  |  | ||||||
|     @Nonnull |     @Nonnull | ||||||
|     @Override |     @Override | ||||||
|  |     @Deprecated | ||||||
|     public TextureAtlasSprite getParticleTexture() |     public TextureAtlasSprite getParticleTexture() | ||||||
|     { |     { | ||||||
|         return m_baseModel.getParticleTexture(); |         return m_baseModel.getParticleTexture(); | ||||||
|   | |||||||
| @@ -12,13 +12,13 @@ import dan200.computercraft.api.turtle.TurtleSide; | |||||||
| import dan200.computercraft.shared.turtle.items.ItemTurtle; | 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.BlockState; | ||||||
| import net.minecraft.client.Minecraft; | import net.minecraft.client.Minecraft; | ||||||
| import net.minecraft.client.renderer.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.LivingEntity; | ||||||
| import net.minecraft.item.ItemStack; | import net.minecraft.item.ItemStack; | ||||||
| import net.minecraft.util.EnumFacing; | import net.minecraft.util.Direction; | ||||||
| import net.minecraft.util.ResourceLocation; | import net.minecraft.util.ResourceLocation; | ||||||
| import net.minecraft.world.World; | import net.minecraft.world.World; | ||||||
| import net.minecraftforge.client.model.data.IModelData; | import net.minecraftforge.client.model.data.IModelData; | ||||||
| @@ -111,7 +111,7 @@ public class TurtleSmartItemModel implements IBakedModel | |||||||
|         { |         { | ||||||
|             @Nonnull |             @Nonnull | ||||||
|             @Override |             @Override | ||||||
|             public IBakedModel getModelWithOverrides( @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 LivingEntity entity ) | ||||||
|             { |             { | ||||||
|                 ItemTurtle turtle = (ItemTurtle) stack.getItem(); |                 ItemTurtle turtle = (ItemTurtle) stack.getItem(); | ||||||
|                 int colour = turtle.getColour( stack ); |                 int colour = turtle.getColour( stack ); | ||||||
| @@ -169,7 +169,7 @@ public class TurtleSmartItemModel implements IBakedModel | |||||||
|     @Nonnull |     @Nonnull | ||||||
|     @Override |     @Override | ||||||
|     @Deprecated |     @Deprecated | ||||||
|     public List<BakedQuad> getQuads( IBlockState state, EnumFacing facing, @Nonnull Random rand ) |     public List<BakedQuad> getQuads( BlockState state, Direction facing, @Nonnull Random rand ) | ||||||
|     { |     { | ||||||
|         return familyModel.getQuads( state, facing, rand ); |         return familyModel.getQuads( state, facing, rand ); | ||||||
|     } |     } | ||||||
| @@ -177,7 +177,7 @@ public class TurtleSmartItemModel implements IBakedModel | |||||||
|     @Nonnull |     @Nonnull | ||||||
|     @Override |     @Override | ||||||
|     @Deprecated |     @Deprecated | ||||||
|     public List<BakedQuad> getQuads( IBlockState state, EnumFacing facing, @Nonnull Random rand, @Nonnull IModelData data ) |     public List<BakedQuad> getQuads( BlockState state, Direction facing, @Nonnull Random rand, @Nonnull IModelData data ) | ||||||
|     { |     { | ||||||
|         return familyModel.getQuads( state, facing, rand, data ); |         return familyModel.getQuads( state, facing, rand, data ); | ||||||
|     } |     } | ||||||
| @@ -202,6 +202,7 @@ public class TurtleSmartItemModel implements IBakedModel | |||||||
|  |  | ||||||
|     @Nonnull |     @Nonnull | ||||||
|     @Override |     @Override | ||||||
|  |     @Deprecated | ||||||
|     public TextureAtlasSprite getParticleTexture() |     public TextureAtlasSprite getParticleTexture() | ||||||
|     { |     { | ||||||
|         return familyModel.getParticleTexture(); |         return familyModel.getParticleTexture(); | ||||||
|   | |||||||
| @@ -160,7 +160,7 @@ public class AddressPredicate | |||||||
|     } |     } | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * Determine whether the given address matches a series of patterns |      * Determine whether the given address matches a series of patterns. | ||||||
|      * |      * | ||||||
|      * @param address The address to check. |      * @param address The address to check. | ||||||
|      * @return Whether it matches any of these patterns. |      * @return Whether it matches any of these patterns. | ||||||
|   | |||||||
| @@ -13,256 +13,106 @@ import javax.annotation.Nullable; | |||||||
| import java.util.Map; | import java.util.Map; | ||||||
|  |  | ||||||
| /** | /** | ||||||
|  * Various helpers for arguments |  * A stub for any mods which depended on this version of the argument helper. | ||||||
|  |  * | ||||||
|  |  * @deprecated Use {@link dan200.computercraft.api.lua.ArgumentHelper}. | ||||||
|  */ |  */ | ||||||
|  | @Deprecated | ||||||
| public final class ArgumentHelper | public final class ArgumentHelper | ||||||
| { | { | ||||||
|     private ArgumentHelper() |     private ArgumentHelper() | ||||||
|     { |     { | ||||||
|         throw new IllegalStateException( "Cannot instantiate singleton " + getClass().getName() ); |  | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @Nonnull |     @Nonnull | ||||||
|     public static String getType( @Nullable Object type ) |     public static String getType( @Nullable Object type ) | ||||||
|     { |     { | ||||||
|         if( type == null ) return "nil"; |         return dan200.computercraft.api.lua.ArgumentHelper.getType( type ); | ||||||
|         if( type instanceof String ) return "string"; |  | ||||||
|         if( type instanceof Boolean ) return "boolean"; |  | ||||||
|         if( type instanceof Number ) return "number"; |  | ||||||
|         if( type instanceof Map ) return "table"; |  | ||||||
|  |  | ||||||
|         Class<?> klass = type.getClass(); |  | ||||||
|         if( klass.isArray() ) |  | ||||||
|         { |  | ||||||
|             StringBuilder name = new StringBuilder(); |  | ||||||
|             while( klass.isArray() ) |  | ||||||
|             { |  | ||||||
|                 name.append( "[]" ); |  | ||||||
|                 klass = klass.getComponentType(); |  | ||||||
|             } |  | ||||||
|             name.insert( 0, klass.getName() ); |  | ||||||
|             return name.toString(); |  | ||||||
|         } |  | ||||||
|         else |  | ||||||
|         { |  | ||||||
|             return klass.getName(); |  | ||||||
|         } |  | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @Nonnull |     @Nonnull | ||||||
|     public static LuaException badArgument( int index, @Nonnull String expected, @Nullable Object actual ) |     public static LuaException badArgument( int index, @Nonnull String expected, @Nullable Object actual ) | ||||||
|     { |     { | ||||||
|         return badArgument( index, expected, getType( actual ) ); |         return dan200.computercraft.api.lua.ArgumentHelper.badArgumentOf( index, expected, actual ); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @Nonnull |     @Nonnull | ||||||
|     public static LuaException badArgument( int index, @Nonnull String expected, @Nonnull String actual ) |     public static LuaException badArgument( int index, @Nonnull String expected, @Nonnull String actual ) | ||||||
|     { |     { | ||||||
|         return new LuaException( "bad argument #" + (index + 1) + " (" + expected + " expected, got " + actual + ")" ); |         return dan200.computercraft.api.lua.ArgumentHelper.badArgument( index, expected, actual ); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     public static double getNumber( @Nonnull Object[] args, int index ) throws LuaException |     public static double getNumber( @Nonnull Object[] args, int index ) throws LuaException | ||||||
|     { |     { | ||||||
|         if( index >= args.length ) throw badArgument( index, "number", "nil" ); |         return dan200.computercraft.api.lua.ArgumentHelper.getDouble( args, index ); | ||||||
|         Object value = args[index]; |  | ||||||
|         if( value instanceof Number ) |  | ||||||
|         { |  | ||||||
|             return ((Number) value).doubleValue(); |  | ||||||
|         } |  | ||||||
|         else |  | ||||||
|         { |  | ||||||
|             throw badArgument( index, "number", value ); |  | ||||||
|         } |  | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     public static int getInt( @Nonnull Object[] args, int index ) throws LuaException |     public static int getInt( @Nonnull Object[] args, int index ) throws LuaException | ||||||
|     { |     { | ||||||
|         return (int) getLong( args, index ); |         return dan200.computercraft.api.lua.ArgumentHelper.getInt( args, index ); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     public static long getLong( @Nonnull Object[] args, int index ) throws LuaException |     public static long getLong( @Nonnull Object[] args, int index ) throws LuaException | ||||||
|     { |     { | ||||||
|         if( index >= args.length ) throw badArgument( index, "number", "nil" ); |         return dan200.computercraft.api.lua.ArgumentHelper.getLong( args, index ); | ||||||
|         Object value = args[index]; |  | ||||||
|         if( value instanceof Number ) |  | ||||||
|         { |  | ||||||
|             return checkReal( index, (Number) value ).longValue(); |  | ||||||
|         } |  | ||||||
|         else |  | ||||||
|         { |  | ||||||
|             throw badArgument( index, "number", value ); |  | ||||||
|         } |  | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     public static double getReal( @Nonnull Object[] args, int index ) throws LuaException |     public static double getReal( @Nonnull Object[] args, int index ) throws LuaException | ||||||
|     { |     { | ||||||
|         return checkReal( index, getNumber( args, index ) ); |         return dan200.computercraft.api.lua.ArgumentHelper.getFiniteDouble( args, index ); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     public static boolean getBoolean( @Nonnull Object[] args, int index ) throws LuaException |     public static boolean getBoolean( @Nonnull Object[] args, int index ) throws LuaException | ||||||
|     { |     { | ||||||
|         if( index >= args.length ) throw badArgument( index, "boolean", "nil" ); |         return dan200.computercraft.api.lua.ArgumentHelper.getBoolean( args, index ); | ||||||
|         Object value = args[index]; |  | ||||||
|         if( value instanceof Boolean ) |  | ||||||
|         { |  | ||||||
|             return (Boolean) value; |  | ||||||
|         } |  | ||||||
|         else |  | ||||||
|         { |  | ||||||
|             throw badArgument( index, "boolean", value ); |  | ||||||
|         } |  | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @Nonnull |     @Nonnull | ||||||
|     public static String getString( @Nonnull Object[] args, int index ) throws LuaException |     public static String getString( @Nonnull Object[] args, int index ) throws LuaException | ||||||
|     { |     { | ||||||
|         if( index >= args.length ) throw badArgument( index, "string", "nil" ); |         return dan200.computercraft.api.lua.ArgumentHelper.getString( args, index ); | ||||||
|         Object value = args[index]; |  | ||||||
|         if( value instanceof String ) |  | ||||||
|         { |  | ||||||
|             return (String) value; |  | ||||||
|         } |  | ||||||
|         else |  | ||||||
|         { |  | ||||||
|             throw badArgument( index, "string", value ); |  | ||||||
|         } |  | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @SuppressWarnings( "unchecked" ) |  | ||||||
|     @Nonnull |     @Nonnull | ||||||
|  |     @SuppressWarnings( "unchecked" ) | ||||||
|     public static Map<Object, Object> getTable( @Nonnull Object[] args, int index ) throws LuaException |     public static Map<Object, Object> getTable( @Nonnull Object[] args, int index ) throws LuaException | ||||||
|     { |     { | ||||||
|         if( index >= args.length ) throw badArgument( index, "table", "nil" ); |         return (Map<Object, Object>) dan200.computercraft.api.lua.ArgumentHelper.getTable( args, index ); | ||||||
|         Object value = args[index]; |  | ||||||
|         if( value instanceof Map ) |  | ||||||
|         { |  | ||||||
|             return (Map<Object, Object>) value; |  | ||||||
|         } |  | ||||||
|         else |  | ||||||
|         { |  | ||||||
|             throw badArgument( index, "table", value ); |  | ||||||
|         } |  | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     public static double optNumber( @Nonnull Object[] args, int index, double def ) throws LuaException |     public static double optNumber( @Nonnull Object[] args, int index, double def ) throws LuaException | ||||||
|     { |     { | ||||||
|         Object value = index < args.length ? args[index] : null; |         return dan200.computercraft.api.lua.ArgumentHelper.optDouble( args, index, def ); | ||||||
|         if( value == null ) |  | ||||||
|         { |  | ||||||
|             return def; |  | ||||||
|         } |  | ||||||
|         else if( value instanceof Number ) |  | ||||||
|         { |  | ||||||
|             return ((Number) value).doubleValue(); |  | ||||||
|         } |  | ||||||
|         else |  | ||||||
|         { |  | ||||||
|             throw badArgument( index, "number", value ); |  | ||||||
|         } |  | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     public static int optInt( @Nonnull Object[] args, int index, int def ) throws LuaException |     public static int optInt( @Nonnull Object[] args, int index, int def ) throws LuaException | ||||||
|     { |     { | ||||||
|         return (int) optLong( args, index, def ); |         return dan200.computercraft.api.lua.ArgumentHelper.optInt( args, index, def ); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     public static long optLong( @Nonnull Object[] args, int index, long def ) throws LuaException |     public static long optLong( @Nonnull Object[] args, int index, long def ) throws LuaException | ||||||
|     { |     { | ||||||
|         Object value = index < args.length ? args[index] : null; |         return dan200.computercraft.api.lua.ArgumentHelper.optLong( args, index, def ); | ||||||
|         if( value == null ) |  | ||||||
|         { |  | ||||||
|             return def; |  | ||||||
|         } |  | ||||||
|         else if( value instanceof Number ) |  | ||||||
|         { |  | ||||||
|             return checkReal( index, (Number) value ).longValue(); |  | ||||||
|         } |  | ||||||
|         else |  | ||||||
|         { |  | ||||||
|             throw badArgument( index, "number", value ); |  | ||||||
|         } |  | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     public static double optReal( @Nonnull Object[] args, int index, double def ) throws LuaException |     public static double optReal( @Nonnull Object[] args, int index, double def ) throws LuaException | ||||||
|     { |     { | ||||||
|         return checkReal( index, optNumber( args, index, def ) ); |         return dan200.computercraft.api.lua.ArgumentHelper.optFiniteDouble( args, index, def ); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     public static boolean optBoolean( @Nonnull Object[] args, int index, boolean def ) throws LuaException |     public static boolean optBoolean( @Nonnull Object[] args, int index, boolean def ) throws LuaException | ||||||
|     { |     { | ||||||
|         Object value = index < args.length ? args[index] : null; |         return dan200.computercraft.api.lua.ArgumentHelper.optBoolean( args, index, def ); | ||||||
|         if( value == null ) |  | ||||||
|         { |  | ||||||
|             return def; |  | ||||||
|         } |  | ||||||
|         else if( value instanceof Boolean ) |  | ||||||
|         { |  | ||||||
|             return (Boolean) value; |  | ||||||
|         } |  | ||||||
|         else |  | ||||||
|         { |  | ||||||
|             throw badArgument( index, "boolean", value ); |  | ||||||
|         } |  | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     public static String optString( @Nonnull Object[] args, int index, String def ) throws LuaException |     public static String optString( @Nonnull Object[] args, int index, String def ) throws LuaException | ||||||
|     { |     { | ||||||
|         Object value = index < args.length ? args[index] : null; |         return dan200.computercraft.api.lua.ArgumentHelper.optString( args, index, def ); | ||||||
|         if( value == null ) |  | ||||||
|         { |  | ||||||
|             return def; |  | ||||||
|         } |  | ||||||
|         else if( value instanceof String ) |  | ||||||
|         { |  | ||||||
|             return (String) value; |  | ||||||
|         } |  | ||||||
|         else |  | ||||||
|         { |  | ||||||
|             throw badArgument( index, "string", value ); |  | ||||||
|         } |  | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @SuppressWarnings( "unchecked" ) |     @SuppressWarnings( "unchecked" ) | ||||||
|     public static Map<Object, Object> optTable( @Nonnull Object[] args, int index, Map<Object, Object> def ) throws LuaException |     public static Map<Object, Object> optTable( @Nonnull Object[] args, int index, Map<Object, Object> def ) throws LuaException | ||||||
|     { |     { | ||||||
|         Object value = index < args.length ? args[index] : null; |         return (Map<Object, Object>) dan200.computercraft.api.lua.ArgumentHelper.optTable( args, index, def ); | ||||||
|         if( value == null ) |  | ||||||
|         { |  | ||||||
|             return def; |  | ||||||
|         } |  | ||||||
|         else if( value instanceof Map ) |  | ||||||
|         { |  | ||||||
|             return (Map<Object, Object>) value; |  | ||||||
|         } |  | ||||||
|         else |  | ||||||
|         { |  | ||||||
|             throw badArgument( index, "table", value ); |  | ||||||
|         } |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     private static Number checkReal( int index, Number value ) throws LuaException |  | ||||||
|     { |  | ||||||
|         checkReal( index, value.doubleValue() ); |  | ||||||
|         return value; |  | ||||||
|     } |  | ||||||
|  |  | ||||||
|     private static double checkReal( int index, double value ) throws LuaException |  | ||||||
|     { |  | ||||||
|         if( Double.isNaN( value ) ) |  | ||||||
|         { |  | ||||||
|             throw badArgument( index, "number", "nan" ); |  | ||||||
|         } |  | ||||||
|         else if( value == Double.POSITIVE_INFINITY ) |  | ||||||
|         { |  | ||||||
|             throw badArgument( index, "number", "inf" ); |  | ||||||
|         } |  | ||||||
|         else if( value == Double.NEGATIVE_INFINITY ) |  | ||||||
|         { |  | ||||||
|             throw badArgument( index, "number", "-inf" ); |  | ||||||
|         } |  | ||||||
|         else |  | ||||||
|         { |  | ||||||
|             return value; |  | ||||||
|         } |  | ||||||
|     } |     } | ||||||
| } | } | ||||||
|   | |||||||
| @@ -27,7 +27,7 @@ import java.util.HashMap; | |||||||
| import java.util.Map; | import java.util.Map; | ||||||
| import java.util.function.Function; | import java.util.function.Function; | ||||||
|  |  | ||||||
| import static dan200.computercraft.core.apis.ArgumentHelper.getString; | import static dan200.computercraft.api.lua.ArgumentHelper.getString; | ||||||
|  |  | ||||||
| public class FSAPI implements ILuaAPI | public class FSAPI implements ILuaAPI | ||||||
| { | { | ||||||
| @@ -43,9 +43,7 @@ public class FSAPI implements ILuaAPI | |||||||
|     @Override |     @Override | ||||||
|     public String[] getNames() |     public String[] getNames() | ||||||
|     { |     { | ||||||
|         return new String[] { |         return new String[] { "fs" }; | ||||||
|             "fs" |  | ||||||
|         }; |  | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @Override |     @Override | ||||||
|   | |||||||
| @@ -23,7 +23,7 @@ import java.util.Collections; | |||||||
| import java.util.Locale; | import java.util.Locale; | ||||||
| import java.util.Map; | import java.util.Map; | ||||||
|  |  | ||||||
| import static dan200.computercraft.core.apis.ArgumentHelper.*; | import static dan200.computercraft.api.lua.ArgumentHelper.*; | ||||||
| import static dan200.computercraft.core.apis.TableHelper.*; | import static dan200.computercraft.core.apis.TableHelper.*; | ||||||
|  |  | ||||||
| public class HTTPAPI implements ILuaAPI | public class HTTPAPI implements ILuaAPI | ||||||
| @@ -42,9 +42,7 @@ public class HTTPAPI implements ILuaAPI | |||||||
|     @Override |     @Override | ||||||
|     public String[] getNames() |     public String[] getNames() | ||||||
|     { |     { | ||||||
|         return new String[] { |         return new String[] { "http" }; | ||||||
|             "http" |  | ||||||
|         }; |  | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @Override |     @Override | ||||||
| @@ -83,6 +81,7 @@ public class HTTPAPI implements ILuaAPI | |||||||
|     } |     } | ||||||
|  |  | ||||||
|     @Override |     @Override | ||||||
|  |     @SuppressWarnings( "resource" ) | ||||||
|     public Object[] callMethod( @Nonnull ILuaContext context, int method, @Nonnull Object[] args ) throws LuaException |     public Object[] callMethod( @Nonnull ILuaContext context, int method, @Nonnull Object[] args ) throws LuaException | ||||||
|     { |     { | ||||||
|         switch( method ) |         switch( method ) | ||||||
| @@ -90,12 +89,12 @@ public class HTTPAPI implements ILuaAPI | |||||||
|             case 0: // request |             case 0: // request | ||||||
|             { |             { | ||||||
|                 String address, postString, requestMethod; |                 String address, postString, requestMethod; | ||||||
|                 Map<Object, Object> headerTable; |                 Map<?, ?> headerTable; | ||||||
|                 boolean binary, redirect; |                 boolean binary, redirect; | ||||||
|  |  | ||||||
|                 if( args.length >= 1 && args[0] instanceof Map ) |                 if( args.length >= 1 && args[0] instanceof Map ) | ||||||
|                 { |                 { | ||||||
|                     Map<?, ?> options = (Map) args[0]; |                     Map<?, ?> options = (Map<?, ?>) args[0]; | ||||||
|                     address = getStringField( options, "url" ); |                     address = getStringField( options, "url" ); | ||||||
|                     postString = optStringField( options, "body", null ); |                     postString = optStringField( options, "body", null ); | ||||||
|                     headerTable = optTableField( options, "headers", Collections.emptyMap() ); |                     headerTable = optTableField( options, "headers", Collections.emptyMap() ); | ||||||
| @@ -135,7 +134,6 @@ public class HTTPAPI implements ILuaAPI | |||||||
|                 try |                 try | ||||||
|                 { |                 { | ||||||
|                     URI uri = HttpRequest.checkUri( address ); |                     URI uri = HttpRequest.checkUri( address ); | ||||||
|  |  | ||||||
|                     HttpRequest request = new HttpRequest( requests, m_apiEnvironment, address, postString, headers, binary, redirect ); |                     HttpRequest request = new HttpRequest( requests, m_apiEnvironment, address, postString, headers, binary, redirect ); | ||||||
|  |  | ||||||
|                     long requestBody = request.body().readableBytes() + HttpRequest.getHeaderSize( headers ); |                     long requestBody = request.body().readableBytes() + HttpRequest.getHeaderSize( headers ); | ||||||
| @@ -174,7 +172,7 @@ public class HTTPAPI implements ILuaAPI | |||||||
|             case 2: // websocket |             case 2: // websocket | ||||||
|             { |             { | ||||||
|                 String address = getString( args, 0 ); |                 String address = getString( args, 0 ); | ||||||
|                 Map<Object, Object> headerTbl = optTable( args, 1, Collections.emptyMap() ); |                 Map<?, ?> headerTbl = optTable( args, 1, Collections.emptyMap() ); | ||||||
|  |  | ||||||
|                 if( !ComputerCraft.http_websocket_enable ) |                 if( !ComputerCraft.http_websocket_enable ) | ||||||
|                 { |                 { | ||||||
|   | |||||||
| @@ -10,6 +10,7 @@ package dan200.computercraft.core.apis; | |||||||
|  * This exists purely to ensure binary compatibility. |  * This exists purely to ensure binary compatibility. | ||||||
|  * |  * | ||||||
|  * @see dan200.computercraft.api.lua.ILuaAPI |  * @see dan200.computercraft.api.lua.ILuaAPI | ||||||
|  |  * @deprecated Use the version in the public API. Only exists for compatibility with CCEmuX. | ||||||
|  */ |  */ | ||||||
| @Deprecated | @Deprecated | ||||||
| public interface ILuaAPI extends dan200.computercraft.api.lua.ILuaAPI | public interface ILuaAPI extends dan200.computercraft.api.lua.ILuaAPI | ||||||
|   | |||||||
| @@ -19,7 +19,7 @@ import java.time.ZonedDateTime; | |||||||
| import java.time.format.DateTimeFormatterBuilder; | import java.time.format.DateTimeFormatterBuilder; | ||||||
| import java.util.*; | import java.util.*; | ||||||
|  |  | ||||||
| import static dan200.computercraft.core.apis.ArgumentHelper.*; | import static dan200.computercraft.api.lua.ArgumentHelper.*; | ||||||
|  |  | ||||||
| public class OSAPI implements ILuaAPI | public class OSAPI implements ILuaAPI | ||||||
| { | { | ||||||
| @@ -36,9 +36,9 @@ public class OSAPI implements ILuaAPI | |||||||
|  |  | ||||||
|     private static class Timer |     private static class Timer | ||||||
|     { |     { | ||||||
|         public int m_ticksLeft; |         int m_ticksLeft; | ||||||
|  |  | ||||||
|         public Timer( int ticksLeft ) |         Timer( int ticksLeft ) | ||||||
|         { |         { | ||||||
|             m_ticksLeft = ticksLeft; |             m_ticksLeft = ticksLeft; | ||||||
|         } |         } | ||||||
| @@ -46,10 +46,10 @@ public class OSAPI implements ILuaAPI | |||||||
|  |  | ||||||
|     private static class Alarm implements Comparable<Alarm> |     private static class Alarm implements Comparable<Alarm> | ||||||
|     { |     { | ||||||
|         public final double m_time; |         final double m_time; | ||||||
|         public final int m_day; |         final int m_day; | ||||||
|  |  | ||||||
|         public Alarm( double time, int day ) |         Alarm( double time, int day ) | ||||||
|         { |         { | ||||||
|             m_time = time; |             m_time = time; | ||||||
|             m_day = day; |             m_day = day; | ||||||
| @@ -78,9 +78,7 @@ public class OSAPI implements ILuaAPI | |||||||
|     @Override |     @Override | ||||||
|     public String[] getNames() |     public String[] getNames() | ||||||
|     { |     { | ||||||
|         return new String[] { |         return new String[] { "os" }; | ||||||
|             "os" |  | ||||||
|         }; |  | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @Override |     @Override | ||||||
| @@ -231,7 +229,7 @@ public class OSAPI implements ILuaAPI | |||||||
|             case 1: |             case 1: | ||||||
|             { |             { | ||||||
|                 // startTimer |                 // startTimer | ||||||
|                 double timer = getReal( args, 0 ); |                 double timer = getFiniteDouble( args, 0 ); | ||||||
|                 synchronized( m_timers ) |                 synchronized( m_timers ) | ||||||
|                 { |                 { | ||||||
|                     m_timers.put( m_nextTimerToken, new Timer( (int) Math.round( timer / 0.05 ) ) ); |                     m_timers.put( m_nextTimerToken, new Timer( (int) Math.round( timer / 0.05 ) ) ); | ||||||
| @@ -241,7 +239,7 @@ public class OSAPI implements ILuaAPI | |||||||
|             case 2: |             case 2: | ||||||
|             { |             { | ||||||
|                 // setAlarm |                 // setAlarm | ||||||
|                 double time = getReal( args, 0 ); |                 double time = getFiniteDouble( args, 0 ); | ||||||
|                 if( time < 0.0 || time >= 24.0 ) |                 if( time < 0.0 || time >= 24.0 ) | ||||||
|                 { |                 { | ||||||
|                     throw new LuaException( "Number out of range" ); |                     throw new LuaException( "Number out of range" ); | ||||||
| @@ -385,9 +383,7 @@ public class OSAPI implements ILuaAPI | |||||||
|                         // Get in-game epoch |                         // Get in-game epoch | ||||||
|                         synchronized( m_alarms ) |                         synchronized( m_alarms ) | ||||||
|                         { |                         { | ||||||
|                             return new Object[] { |                             return new Object[] { m_day * 86400000 + (int) (m_time * 3600000.0f) }; | ||||||
|                                 m_day * 86400000 + (int) (m_time * 3600000.0f) |  | ||||||
|                             }; |  | ||||||
|                         } |                         } | ||||||
|                     default: |                     default: | ||||||
|                         throw new LuaException( "Unsupported operation" ); |                         throw new LuaException( "Unsupported operation" ); | ||||||
| @@ -401,7 +397,6 @@ public class OSAPI implements ILuaAPI | |||||||
|                 Instant instant = Instant.ofEpochSecond( time ); |                 Instant instant = Instant.ofEpochSecond( time ); | ||||||
|                 ZonedDateTime date; |                 ZonedDateTime date; | ||||||
|                 ZoneOffset offset; |                 ZoneOffset offset; | ||||||
|                 boolean isDst; |  | ||||||
|                 if( format.startsWith( "!" ) ) |                 if( format.startsWith( "!" ) ) | ||||||
|                 { |                 { | ||||||
|                     offset = ZoneOffset.UTC; |                     offset = ZoneOffset.UTC; | ||||||
|   | |||||||
| @@ -22,7 +22,7 @@ import java.util.Collections; | |||||||
| import java.util.HashMap; | import java.util.HashMap; | ||||||
| import java.util.Map; | import java.util.Map; | ||||||
|  |  | ||||||
| import static dan200.computercraft.core.apis.ArgumentHelper.getString; | import static dan200.computercraft.api.lua.ArgumentHelper.getString; | ||||||
|  |  | ||||||
| public class PeripheralAPI implements ILuaAPI, IAPIEnvironment.IPeripheralChangeListener | public class PeripheralAPI implements ILuaAPI, IAPIEnvironment.IPeripheralChangeListener | ||||||
| { | { | ||||||
| @@ -36,7 +36,7 @@ public class PeripheralAPI implements ILuaAPI, IAPIEnvironment.IPeripheralChange | |||||||
|         private Map<String, Integer> m_methodMap; |         private Map<String, Integer> m_methodMap; | ||||||
|         private boolean m_attached; |         private boolean m_attached; | ||||||
|  |  | ||||||
|         public PeripheralWrapper( IPeripheral peripheral, String side ) |         PeripheralWrapper( IPeripheral peripheral, String side ) | ||||||
|         { |         { | ||||||
|             super( m_environment ); |             super( m_environment ); | ||||||
|             m_side = side; |             m_side = side; | ||||||
| @@ -282,9 +282,7 @@ public class PeripheralAPI implements ILuaAPI, IAPIEnvironment.IPeripheralChange | |||||||
|     @Override |     @Override | ||||||
|     public String[] getNames() |     public String[] getNames() | ||||||
|     { |     { | ||||||
|         return new String[] { |         return new String[] { "peripheral" }; | ||||||
|             "peripheral" |  | ||||||
|         }; |  | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @Override |     @Override | ||||||
| @@ -326,7 +324,7 @@ public class PeripheralAPI implements ILuaAPI, IAPIEnvironment.IPeripheralChange | |||||||
|             "isPresent", |             "isPresent", | ||||||
|             "getType", |             "getType", | ||||||
|             "getMethods", |             "getMethods", | ||||||
|             "call" |             "call", | ||||||
|         }; |         }; | ||||||
|     } |     } | ||||||
|  |  | ||||||
| @@ -356,7 +354,6 @@ public class PeripheralAPI implements ILuaAPI, IAPIEnvironment.IPeripheralChange | |||||||
|                 ComputerSide side = ComputerSide.valueOfInsensitive( getString( args, 0 ) ); |                 ComputerSide side = ComputerSide.valueOfInsensitive( getString( args, 0 ) ); | ||||||
|                 if( side != null ) |                 if( side != null ) | ||||||
|                 { |                 { | ||||||
|                     String type = null; |  | ||||||
|                     synchronized( m_peripherals ) |                     synchronized( m_peripherals ) | ||||||
|                     { |                     { | ||||||
|                         PeripheralWrapper p = m_peripherals[side.ordinal()]; |                         PeripheralWrapper p = m_peripherals[side.ordinal()]; | ||||||
|   | |||||||
| @@ -15,7 +15,7 @@ import javax.annotation.Nonnull; | |||||||
| import java.util.HashMap; | import java.util.HashMap; | ||||||
| import java.util.Map; | import java.util.Map; | ||||||
|  |  | ||||||
| import static dan200.computercraft.core.apis.ArgumentHelper.*; | import static dan200.computercraft.api.lua.ArgumentHelper.*; | ||||||
|  |  | ||||||
| public class RedstoneAPI implements ILuaAPI | public class RedstoneAPI implements ILuaAPI | ||||||
| { | { | ||||||
| @@ -29,9 +29,7 @@ public class RedstoneAPI implements ILuaAPI | |||||||
|     @Override |     @Override | ||||||
|     public String[] getNames() |     public String[] getNames() | ||||||
|     { |     { | ||||||
|         return new String[] { |         return new String[] { "rs", "redstone" }; | ||||||
|             "rs", "redstone" |  | ||||||
|         }; |  | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @Nonnull |     @Nonnull | ||||||
|   | |||||||
| @@ -6,14 +6,17 @@ | |||||||
|  |  | ||||||
| package dan200.computercraft.core.apis; | package dan200.computercraft.core.apis; | ||||||
|  |  | ||||||
|  | import dan200.computercraft.api.lua.ArgumentHelper; | ||||||
| import dan200.computercraft.api.lua.LuaException; | import dan200.computercraft.api.lua.LuaException; | ||||||
|  |  | ||||||
| import javax.annotation.Nonnull; | import javax.annotation.Nonnull; | ||||||
| import javax.annotation.Nullable; | import javax.annotation.Nullable; | ||||||
| import java.util.Map; | import java.util.Map; | ||||||
|  |  | ||||||
|  | import static dan200.computercraft.api.lua.ArgumentHelper.getNumericType; | ||||||
|  |  | ||||||
| /** | /** | ||||||
|  * Various helpers for tables |  * Various helpers for tables. | ||||||
|  */ |  */ | ||||||
| public final class TableHelper | public final class TableHelper | ||||||
| { | { | ||||||
| @@ -200,21 +203,7 @@ public final class TableHelper | |||||||
|  |  | ||||||
|     private static double checkReal( @Nonnull String key, double value ) throws LuaException |     private static double checkReal( @Nonnull String key, double value ) throws LuaException | ||||||
|     { |     { | ||||||
|         if( Double.isNaN( value ) ) |         if( !Double.isFinite( value ) ) throw badKey( key, "number", getNumericType( value ) ); | ||||||
|         { |         return value; | ||||||
|             throw badKey( key, "number", "nan" ); |  | ||||||
|         } |  | ||||||
|         else if( value == Double.POSITIVE_INFINITY ) |  | ||||||
|         { |  | ||||||
|             throw badKey( key, "number", "inf" ); |  | ||||||
|         } |  | ||||||
|         else if( value == Double.NEGATIVE_INFINITY ) |  | ||||||
|         { |  | ||||||
|             throw badKey( key, "number", "-inf" ); |  | ||||||
|         } |  | ||||||
|         else |  | ||||||
|         { |  | ||||||
|             return value; |  | ||||||
|         } |  | ||||||
|     } |     } | ||||||
| } | } | ||||||
|   | |||||||
| @@ -17,7 +17,7 @@ import org.apache.commons.lang3.ArrayUtils; | |||||||
|  |  | ||||||
| import javax.annotation.Nonnull; | import javax.annotation.Nonnull; | ||||||
|  |  | ||||||
| import static dan200.computercraft.core.apis.ArgumentHelper.*; | import static dan200.computercraft.api.lua.ArgumentHelper.*; | ||||||
|  |  | ||||||
| public class TermAPI implements ILuaAPI | public class TermAPI implements ILuaAPI | ||||||
| { | { | ||||||
| @@ -33,9 +33,7 @@ public class TermAPI implements ILuaAPI | |||||||
|     @Override |     @Override | ||||||
|     public String[] getNames() |     public String[] getNames() | ||||||
|     { |     { | ||||||
|         return new String[] { |         return new String[] { "term" }; | ||||||
|             "term" |  | ||||||
|         }; |  | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @Nonnull |     @Nonnull | ||||||
| @@ -89,9 +87,7 @@ public class TermAPI implements ILuaAPI | |||||||
|  |  | ||||||
|     public static Object[] encodeColour( int colour ) throws LuaException |     public static Object[] encodeColour( int colour ) throws LuaException | ||||||
|     { |     { | ||||||
|         return new Object[] { |         return new Object[] { 1 << colour }; | ||||||
|             1 << colour |  | ||||||
|         }; |  | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     public static void setColour( Terminal terminal, int colour, double r, double g, double b ) |     public static void setColour( Terminal terminal, int colour, double r, double g, double b ) | ||||||
| @@ -246,9 +242,9 @@ public class TermAPI implements ILuaAPI | |||||||
|                 } |                 } | ||||||
|                 else |                 else | ||||||
|                 { |                 { | ||||||
|                     double r = getReal( args, 1 ); |                     double r = getFiniteDouble( args, 1 ); | ||||||
|                     double g = getReal( args, 2 ); |                     double g = getFiniteDouble( args, 2 ); | ||||||
|                     double b = getReal( args, 3 ); |                     double b = getFiniteDouble( args, 3 ); | ||||||
|                     setColour( m_terminal, colour, r, g, b ); |                     setColour( m_terminal, colour, r, g, b ); | ||||||
|                 } |                 } | ||||||
|                 return null; |                 return null; | ||||||
|   | |||||||
| @@ -21,8 +21,8 @@ import java.util.ArrayList; | |||||||
| import java.util.Arrays; | import java.util.Arrays; | ||||||
| import java.util.List; | import java.util.List; | ||||||
|  |  | ||||||
| import static dan200.computercraft.core.apis.ArgumentHelper.getInt; | import static dan200.computercraft.api.lua.ArgumentHelper.getInt; | ||||||
| import static dan200.computercraft.core.apis.ArgumentHelper.optBoolean; | import static dan200.computercraft.api.lua.ArgumentHelper.optBoolean; | ||||||
|  |  | ||||||
| public class BinaryReadableHandle extends HandleGeneric | public class BinaryReadableHandle extends HandleGeneric | ||||||
| { | { | ||||||
| @@ -212,6 +212,7 @@ public class BinaryReadableHandle extends HandleGeneric | |||||||
|                 } |                 } | ||||||
|             } |             } | ||||||
|             case 3: // close |             case 3: // close | ||||||
|  |                 checkOpen(); | ||||||
|                 close(); |                 close(); | ||||||
|                 return null; |                 return null; | ||||||
|             case 4: // seek |             case 4: // seek | ||||||
|   | |||||||
| @@ -7,9 +7,9 @@ | |||||||
| package dan200.computercraft.core.apis.handles; | package dan200.computercraft.core.apis.handles; | ||||||
|  |  | ||||||
| import com.google.common.collect.ObjectArrays; | import com.google.common.collect.ObjectArrays; | ||||||
|  | import dan200.computercraft.api.lua.ArgumentHelper; | ||||||
| import dan200.computercraft.api.lua.ILuaContext; | import dan200.computercraft.api.lua.ILuaContext; | ||||||
| import dan200.computercraft.api.lua.LuaException; | import dan200.computercraft.api.lua.LuaException; | ||||||
| import dan200.computercraft.core.apis.ArgumentHelper; |  | ||||||
| import dan200.computercraft.shared.util.StringUtil; | import dan200.computercraft.shared.util.StringUtil; | ||||||
|  |  | ||||||
| import javax.annotation.Nonnull; | import javax.annotation.Nonnull; | ||||||
| @@ -73,7 +73,7 @@ public class BinaryWritableHandle extends HandleGeneric | |||||||
|                     } |                     } | ||||||
|                     else |                     else | ||||||
|                     { |                     { | ||||||
|                         throw ArgumentHelper.badArgument( 0, "string or number", args.length > 0 ? args[0] : null ); |                         throw ArgumentHelper.badArgumentOf( 0, "string or number", args.length > 0 ? args[0] : null ); | ||||||
|                     } |                     } | ||||||
|                     return null; |                     return null; | ||||||
|                 } |                 } | ||||||
| @@ -95,6 +95,7 @@ public class BinaryWritableHandle extends HandleGeneric | |||||||
|                     return null; |                     return null; | ||||||
|                 } |                 } | ||||||
|             case 2: // close |             case 2: // close | ||||||
|  |                 checkOpen(); | ||||||
|                 close(); |                 close(); | ||||||
|                 return null; |                 return null; | ||||||
|             case 3: // seek |             case 3: // seek | ||||||
|   | |||||||
| @@ -20,8 +20,8 @@ import java.nio.charset.CharsetDecoder; | |||||||
| import java.nio.charset.CodingErrorAction; | import java.nio.charset.CodingErrorAction; | ||||||
| import java.nio.charset.StandardCharsets; | import java.nio.charset.StandardCharsets; | ||||||
|  |  | ||||||
| import static dan200.computercraft.core.apis.ArgumentHelper.optBoolean; | import static dan200.computercraft.api.lua.ArgumentHelper.optBoolean; | ||||||
| import static dan200.computercraft.core.apis.ArgumentHelper.optInt; | import static dan200.computercraft.api.lua.ArgumentHelper.optInt; | ||||||
|  |  | ||||||
| public class EncodedReadableHandle extends HandleGeneric | public class EncodedReadableHandle extends HandleGeneric | ||||||
| { | { | ||||||
| @@ -152,6 +152,7 @@ public class EncodedReadableHandle extends HandleGeneric | |||||||
|                     return null; |                     return null; | ||||||
|                 } |                 } | ||||||
|             case 3: // close |             case 3: // close | ||||||
|  |                 checkOpen(); | ||||||
|                 close(); |                 close(); | ||||||
|                 return null; |                 return null; | ||||||
|             default: |             default: | ||||||
|   | |||||||
| @@ -93,6 +93,7 @@ public class EncodedWritableHandle extends HandleGeneric | |||||||
|                     return null; |                     return null; | ||||||
|                 } |                 } | ||||||
|             case 3: // close |             case 3: // close | ||||||
|  |                 checkOpen(); | ||||||
|                 close(); |                 close(); | ||||||
|                 return null; |                 return null; | ||||||
|             default: |             default: | ||||||
|   | |||||||
| @@ -16,8 +16,8 @@ import java.io.IOException; | |||||||
| import java.nio.channels.Channel; | import java.nio.channels.Channel; | ||||||
| import java.nio.channels.SeekableByteChannel; | import java.nio.channels.SeekableByteChannel; | ||||||
|  |  | ||||||
| import static dan200.computercraft.core.apis.ArgumentHelper.optLong; | import static dan200.computercraft.api.lua.ArgumentHelper.optLong; | ||||||
| import static dan200.computercraft.core.apis.ArgumentHelper.optString; | import static dan200.computercraft.api.lua.ArgumentHelper.optString; | ||||||
|  |  | ||||||
| public abstract class HandleGeneric implements ILuaObject | public abstract class HandleGeneric implements ILuaObject | ||||||
| { | { | ||||||
| @@ -37,12 +37,17 @@ public abstract class HandleGeneric implements ILuaObject | |||||||
|     protected final void close() |     protected final void close() | ||||||
|     { |     { | ||||||
|         m_open = false; |         m_open = false; | ||||||
|         IoUtil.closeQuietly( m_closable ); |  | ||||||
|         m_closable = null; |         Closeable closeable = m_closable; | ||||||
|  |         if( closeable != null ) | ||||||
|  |         { | ||||||
|  |             IoUtil.closeQuietly( closeable ); | ||||||
|  |             m_closable = null; | ||||||
|  |         } | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * Shared implementation for various file handle types |      * Shared implementation for various file handle types. | ||||||
|      * |      * | ||||||
|      * @param channel The channel to seek in |      * @param channel The channel to seek in | ||||||
|      * @param args    The Lua arguments to process, like Lua's {@code file:seek}. |      * @param args    The Lua arguments to process, like Lua's {@code file:seek}. | ||||||
|   | |||||||
| @@ -99,7 +99,7 @@ public final class NetworkUtils | |||||||
|     } |     } | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * Checks a host is allowed |      * Checks a host is allowed. | ||||||
|      * |      * | ||||||
|      * @param host The domain to check against |      * @param host The domain to check against | ||||||
|      * @throws HTTPRequestException If the host is not permitted. |      * @throws HTTPRequestException If the host is not permitted. | ||||||
|   | |||||||
| @@ -20,6 +20,8 @@ import java.util.function.Consumer; | |||||||
|  |  | ||||||
| /** | /** | ||||||
|  * A holder for one or more resources, with a lifetime. |  * A holder for one or more resources, with a lifetime. | ||||||
|  |  * | ||||||
|  |  * @param <T> The type of this resource. Should be the class extending from {@link Resource}. | ||||||
|  */ |  */ | ||||||
| public abstract class Resource<T extends Resource<T>> implements Closeable | public abstract class Resource<T extends Resource<T>> implements Closeable | ||||||
| { | { | ||||||
| @@ -42,8 +44,9 @@ public abstract class Resource<T extends Resource<T>> implements Closeable | |||||||
|     } |     } | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * Checks if this has been cancelled. If so, it'll clean up any |      * Checks if this has been cancelled. If so, it'll clean up any existing resources and cancel any pending futures. | ||||||
|      * existing resources and cancel any pending futures. |      * | ||||||
|  |      * @return Whether this resource has been closed. | ||||||
|      */ |      */ | ||||||
|     public final boolean checkClosed() |     public final boolean checkClosed() | ||||||
|     { |     { | ||||||
| @@ -72,13 +75,15 @@ public abstract class Resource<T extends Resource<T>> implements Closeable | |||||||
|      */ |      */ | ||||||
|     protected void dispose() |     protected void dispose() | ||||||
|     { |     { | ||||||
|         @SuppressWarnings( "unchecked" ) T thisT = (T) this; |         @SuppressWarnings( "unchecked" ) | ||||||
|  |         T thisT = (T) this; | ||||||
|         limiter.release( thisT ); |         limiter.release( thisT ); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * Create a {@link WeakReference} which will close {@code this} when collected. |      * Create a {@link WeakReference} which will close {@code this} when collected. | ||||||
|      * |      * | ||||||
|  |      * @param <R>    The object we are wrapping in a reference. | ||||||
|      * @param object The object to reference to |      * @param object The object to reference to | ||||||
|      * @return The weak reference. |      * @return The weak reference. | ||||||
|      */ |      */ | ||||||
| @@ -95,7 +100,8 @@ public abstract class Resource<T extends Resource<T>> implements Closeable | |||||||
|  |  | ||||||
|     public boolean queue( Consumer<T> task ) |     public boolean queue( Consumer<T> task ) | ||||||
|     { |     { | ||||||
|         @SuppressWarnings( "unchecked" ) T thisT = (T) this; |         @SuppressWarnings( "unchecked" ) | ||||||
|  |         T thisT = (T) this; | ||||||
|         return limiter.queue( thisT, () -> task.accept( thisT ) ); |         return limiter.queue( thisT, () -> task.accept( thisT ) ); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|   | |||||||
| @@ -14,6 +14,8 @@ import java.util.function.Supplier; | |||||||
|  |  | ||||||
| /** | /** | ||||||
|  * A collection of {@link Resource}s, with an upper bound on capacity. |  * A collection of {@link Resource}s, with an upper bound on capacity. | ||||||
|  |  * | ||||||
|  |  * @param <T> The type of the resource this group manages. | ||||||
|  */ |  */ | ||||||
| public class ResourceGroup<T extends Resource<T>> | public class ResourceGroup<T extends Resource<T>> | ||||||
| { | { | ||||||
|   | |||||||
| @@ -12,6 +12,8 @@ import java.util.function.Supplier; | |||||||
|  |  | ||||||
| /** | /** | ||||||
|  * A {@link ResourceGroup} which will queue items when the group at capacity. |  * A {@link ResourceGroup} which will queue items when the group at capacity. | ||||||
|  |  * | ||||||
|  |  * @param <T> The type of the resource this queue manages. | ||||||
|  */ |  */ | ||||||
| public class ResourceQueue<T extends Resource<T>> extends ResourceGroup<T> | public class ResourceQueue<T extends Resource<T>> extends ResourceGroup<T> | ||||||
| { | { | ||||||
|   | |||||||
| @@ -40,7 +40,7 @@ import java.util.concurrent.TimeUnit; | |||||||
| import java.util.concurrent.atomic.AtomicInteger; | import java.util.concurrent.atomic.AtomicInteger; | ||||||
|  |  | ||||||
| /** | /** | ||||||
|  * Represents one or more |  * Represents an in-progress HTTP request. | ||||||
|  */ |  */ | ||||||
| public class HttpRequest extends Resource<HttpRequest> | public class HttpRequest extends Resource<HttpRequest> | ||||||
| { | { | ||||||
|   | |||||||
| @@ -226,7 +226,11 @@ public final class HttpRequestHandler extends SimpleChannelInboundHandler<HttpOb | |||||||
|     } |     } | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * Determine the redirect from this response |      * Determine the redirect from this response. | ||||||
|  |      * | ||||||
|  |      * @param status  The status of the HTTP response. | ||||||
|  |      * @param headers The headers of the HTTP response. | ||||||
|  |      * @return The URI to redirect to, or {@code null} if no redirect should occur. | ||||||
|      */ |      */ | ||||||
|     private URI getRedirect( HttpResponseStatus status, HttpHeaders headers ) |     private URI getRedirect( HttpResponseStatus status, HttpHeaders headers ) | ||||||
|     { |     { | ||||||
|   | |||||||
| @@ -23,7 +23,7 @@ import javax.annotation.Nullable; | |||||||
| import java.io.Closeable; | import java.io.Closeable; | ||||||
| import java.util.Arrays; | import java.util.Arrays; | ||||||
|  |  | ||||||
| import static dan200.computercraft.core.apis.ArgumentHelper.optBoolean; | import static dan200.computercraft.api.lua.ArgumentHelper.optBoolean; | ||||||
| import static dan200.computercraft.core.apis.http.websocket.Websocket.CLOSE_EVENT; | import static dan200.computercraft.core.apis.http.websocket.Websocket.CLOSE_EVENT; | ||||||
| import static dan200.computercraft.core.apis.http.websocket.Websocket.MESSAGE_EVENT; | import static dan200.computercraft.core.apis.http.websocket.Websocket.MESSAGE_EVENT; | ||||||
|  |  | ||||||
|   | |||||||
| @@ -56,9 +56,6 @@ final class ComputerExecutor | |||||||
| { | { | ||||||
|     private static final int QUEUE_LIMIT = 256; |     private static final int QUEUE_LIMIT = 256; | ||||||
|  |  | ||||||
|     private static IMount romMount; |  | ||||||
|     private static final Object romMountLock = new Object(); |  | ||||||
|  |  | ||||||
|     private final Computer computer; |     private final Computer computer; | ||||||
|     private final List<ILuaAPI> apis = new ArrayList<>(); |     private final List<ILuaAPI> apis = new ArrayList<>(); | ||||||
|     final TimeoutState timeout = new TimeoutState(); |     final TimeoutState timeout = new TimeoutState(); | ||||||
| @@ -268,7 +265,7 @@ final class ComputerExecutor | |||||||
|     } |     } | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * Queue an event if the computer is on |      * Queue an event if the computer is on. | ||||||
|      * |      * | ||||||
|      * @param event The event's name |      * @param event The event's name | ||||||
|      * @param args  The event's arguments |      * @param args  The event's arguments | ||||||
| @@ -329,16 +326,10 @@ final class ComputerExecutor | |||||||
|  |  | ||||||
|     private IMount getRomMount() |     private IMount getRomMount() | ||||||
|     { |     { | ||||||
|         if( romMount != null ) return romMount; |         return computer.getComputerEnvironment().createResourceMount( "computercraft", "lua/rom" ); | ||||||
|  |  | ||||||
|         synchronized( romMountLock ) |  | ||||||
|         { |  | ||||||
|             if( romMount != null ) return romMount; |  | ||||||
|             return romMount = computer.getComputerEnvironment().createResourceMount( "computercraft", "lua/rom" ); |  | ||||||
|         } |  | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     IWritableMount getRootMount() |     private IWritableMount getRootMount() | ||||||
|     { |     { | ||||||
|         if( rootMount == null ) |         if( rootMount == null ) | ||||||
|         { |         { | ||||||
|   | |||||||
| @@ -6,11 +6,13 @@ | |||||||
|  |  | ||||||
| package dan200.computercraft.core.computer; | package dan200.computercraft.core.computer; | ||||||
|  |  | ||||||
|  | import net.minecraft.util.Direction; | ||||||
|  |  | ||||||
| import javax.annotation.Nonnull; | import javax.annotation.Nonnull; | ||||||
| import javax.annotation.Nullable; | import javax.annotation.Nullable; | ||||||
|  |  | ||||||
| /** | /** | ||||||
|  * A side on a computer. Unlike {@link net.minecraft.util.EnumFacing}, this is relative to the direction the computer is |  * A side on a computer. Unlike {@link Direction}, this is relative to the direction the computer is | ||||||
|  * facing.. |  * facing.. | ||||||
|  */ |  */ | ||||||
| public enum ComputerSide | public enum ComputerSide | ||||||
| @@ -30,7 +32,10 @@ public enum ComputerSide | |||||||
|  |  | ||||||
|     private final String name; |     private final String name; | ||||||
|  |  | ||||||
|     ComputerSide( String name ) {this.name = name;} |     ComputerSide( String name ) | ||||||
|  |     { | ||||||
|  |         this.name = name; | ||||||
|  |     } | ||||||
|  |  | ||||||
|     @Nonnull |     @Nonnull | ||||||
|     public static ComputerSide valueOf( int side ) |     public static ComputerSide valueOf( int side ) | ||||||
|   | |||||||
| @@ -50,7 +50,7 @@ import static dan200.computercraft.core.computer.TimeoutState.TIMEOUT; | |||||||
| public final class ComputerThread | public final class ComputerThread | ||||||
| { | { | ||||||
|     /** |     /** | ||||||
|      * How often the computer thread monitor should run, in milliseconds |      * How often the computer thread monitor should run, in milliseconds. | ||||||
|      * |      * | ||||||
|      * @see Monitor |      * @see Monitor | ||||||
|      */ |      */ | ||||||
| @@ -83,7 +83,7 @@ public final class ComputerThread | |||||||
|     private static final Object threadLock = new Object(); |     private static final Object threadLock = new Object(); | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * Whether the computer thread system is currently running |      * Whether the computer thread system is currently running. | ||||||
|      */ |      */ | ||||||
|     private static volatile boolean running = false; |     private static volatile boolean running = false; | ||||||
|  |  | ||||||
| @@ -105,7 +105,7 @@ public final class ComputerThread | |||||||
|     private static final Condition hasWork = computerLock.newCondition(); |     private static final Condition hasWork = computerLock.newCondition(); | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * Active queues to execute |      * Active queues to execute. | ||||||
|      */ |      */ | ||||||
|     private static final TreeSet<ComputerExecutor> computerQueue = new TreeSet<>( ( a, b ) -> { |     private static final TreeSet<ComputerExecutor> computerQueue = new TreeSet<>( ( a, b ) -> { | ||||||
|         if( a == b ) return 0; // Should never happen, but let's be consistent here |         if( a == b ) return 0; // Should never happen, but let's be consistent here | ||||||
| @@ -126,7 +126,7 @@ public final class ComputerThread | |||||||
|     private ComputerThread() {} |     private ComputerThread() {} | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * Start the computer thread |      * Start the computer thread. | ||||||
|      */ |      */ | ||||||
|     static void start() |     static void start() | ||||||
|     { |     { | ||||||
| @@ -194,7 +194,7 @@ public final class ComputerThread | |||||||
|     } |     } | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * Mark a computer as having work, enqueuing it on the thread |      * Mark a computer as having work, enqueuing it on the thread. | ||||||
|      * |      * | ||||||
|      * You must be holding {@link ComputerExecutor}'s {@code queueLock} when calling this method - it should only |      * You must be holding {@link ComputerExecutor}'s {@code queueLock} when calling this method - it should only | ||||||
|      * be called from {@code enqueue}. |      * be called from {@code enqueue}. | ||||||
| @@ -244,6 +244,8 @@ public final class ComputerThread | |||||||
|      * {@link #minimumVirtualRuntime} based on the current tasks. |      * {@link #minimumVirtualRuntime} based on the current tasks. | ||||||
|      * |      * | ||||||
|      * This is called before queueing tasks, to ensure that {@link #minimumVirtualRuntime} is up-to-date. |      * This is called before queueing tasks, to ensure that {@link #minimumVirtualRuntime} is up-to-date. | ||||||
|  |      * | ||||||
|  |      * @param current The machine which we updating runtimes from. | ||||||
|      */ |      */ | ||||||
|     private static void updateRuntimes( @Nullable ComputerExecutor current ) |     private static void updateRuntimes( @Nullable ComputerExecutor current ) | ||||||
|     { |     { | ||||||
| @@ -321,7 +323,7 @@ public final class ComputerThread | |||||||
|     } |     } | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * The scaled period for a single task |      * The scaled period for a single task. | ||||||
|      * |      * | ||||||
|      * @return The scaled period for the task |      * @return The scaled period for the task | ||||||
|      * @see #DEFAULT_LATENCY |      * @see #DEFAULT_LATENCY | ||||||
| @@ -336,7 +338,7 @@ public final class ComputerThread | |||||||
|     } |     } | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * Determine if the thread has computers queued up |      * Determine if the thread has computers queued up. | ||||||
|      * |      * | ||||||
|      * @return If we have work queued up. |      * @return If we have work queued up. | ||||||
|      */ |      */ | ||||||
|   | |||||||
| @@ -36,12 +36,12 @@ import java.util.concurrent.TimeUnit; | |||||||
| public final class TimeoutState | public final class TimeoutState | ||||||
| { | { | ||||||
|     /** |     /** | ||||||
|      * The total time a task is allowed to run before aborting in nanoseconds |      * The total time a task is allowed to run before aborting in nanoseconds. | ||||||
|      */ |      */ | ||||||
|     static final long TIMEOUT = TimeUnit.MILLISECONDS.toNanos( 7000 ); |     static final long TIMEOUT = TimeUnit.MILLISECONDS.toNanos( 7000 ); | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * The time the task is allowed to run after each abort in nanoseconds |      * The time the task is allowed to run after each abort in nanoseconds. | ||||||
|      */ |      */ | ||||||
|     static final long ABORT_TIMEOUT = TimeUnit.MILLISECONDS.toNanos( 1500 ); |     static final long ABORT_TIMEOUT = TimeUnit.MILLISECONDS.toNanos( 1500 ); | ||||||
|  |  | ||||||
| @@ -111,6 +111,8 @@ public final class TimeoutState | |||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * If the machine should be passively aborted. |      * If the machine should be passively aborted. | ||||||
|  |      * | ||||||
|  |      * @return {@code true} if we should throw a timeout error. | ||||||
|      */ |      */ | ||||||
|     public boolean isSoftAborted() |     public boolean isSoftAborted() | ||||||
|     { |     { | ||||||
| @@ -118,7 +120,9 @@ public final class TimeoutState | |||||||
|     } |     } | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * If the machine should be forcibly aborted. |      * Determine if the machine should be forcibly aborted. | ||||||
|  |      * | ||||||
|  |      * @return {@code true} if the machine should be forcibly shut down. | ||||||
|      */ |      */ | ||||||
|     public boolean isHardAborted() |     public boolean isHardAborted() | ||||||
|     { |     { | ||||||
| @@ -146,7 +150,7 @@ public final class TimeoutState | |||||||
|     } |     } | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * Pauses the cumulative time, to be resumed by {@link #startTimer()} |      * Pauses the cumulative time, to be resumed by {@link #startTimer()}. | ||||||
|      * |      * | ||||||
|      * @see #nanoCumulative() |      * @see #nanoCumulative() | ||||||
|      */ |      */ | ||||||
|   | |||||||
| @@ -6,6 +6,7 @@ | |||||||
|  |  | ||||||
| package dan200.computercraft.core.filesystem; | package dan200.computercraft.core.filesystem; | ||||||
|  |  | ||||||
|  | import dan200.computercraft.api.filesystem.FileOperationException; | ||||||
| import dan200.computercraft.api.filesystem.IMount; | import dan200.computercraft.api.filesystem.IMount; | ||||||
|  |  | ||||||
| import javax.annotation.Nonnull; | import javax.annotation.Nonnull; | ||||||
| @@ -95,7 +96,7 @@ public class ComboMount implements IMount | |||||||
|         } |         } | ||||||
|         else |         else | ||||||
|         { |         { | ||||||
|             throw new IOException( "/" + path + ": Not a directory" ); |             throw new FileOperationException( path, "Not a directory" ); | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  |  | ||||||
| @@ -110,7 +111,7 @@ public class ComboMount implements IMount | |||||||
|                 return part.getSize( path ); |                 return part.getSize( path ); | ||||||
|             } |             } | ||||||
|         } |         } | ||||||
|         throw new IOException( "/" + path + ": No such file" ); |         throw new FileOperationException( path, "No such file" ); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @Nonnull |     @Nonnull | ||||||
| @@ -126,7 +127,7 @@ public class ComboMount implements IMount | |||||||
|                 return part.openForRead( path ); |                 return part.openForRead( path ); | ||||||
|             } |             } | ||||||
|         } |         } | ||||||
|         throw new IOException( "/" + path + ": No such file" ); |         throw new FileOperationException( path, "No such file" ); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @Nonnull |     @Nonnull | ||||||
| @@ -141,6 +142,6 @@ public class ComboMount implements IMount | |||||||
|                 return part.openChannelForRead( path ); |                 return part.openChannelForRead( path ); | ||||||
|             } |             } | ||||||
|         } |         } | ||||||
|         throw new IOException( "/" + path + ": No such file" ); |         throw new FileOperationException( path, "No such file" ); | ||||||
|     } |     } | ||||||
| } | } | ||||||
|   | |||||||
| @@ -6,6 +6,7 @@ | |||||||
|  |  | ||||||
| package dan200.computercraft.core.filesystem; | package dan200.computercraft.core.filesystem; | ||||||
|  |  | ||||||
|  | import dan200.computercraft.api.filesystem.FileOperationException; | ||||||
| import dan200.computercraft.api.filesystem.IMount; | import dan200.computercraft.api.filesystem.IMount; | ||||||
|  |  | ||||||
| import javax.annotation.Nonnull; | import javax.annotation.Nonnull; | ||||||
| @@ -44,7 +45,7 @@ public class EmptyMount implements IMount | |||||||
|     @Deprecated |     @Deprecated | ||||||
|     public InputStream openForRead( @Nonnull String path ) throws IOException |     public InputStream openForRead( @Nonnull String path ) throws IOException | ||||||
|     { |     { | ||||||
|         throw new IOException( "/" + path + ": No such file" ); |         throw new FileOperationException( path, "No such file" ); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @Nonnull |     @Nonnull | ||||||
| @@ -52,6 +53,6 @@ public class EmptyMount implements IMount | |||||||
|     @Deprecated |     @Deprecated | ||||||
|     public ReadableByteChannel openChannelForRead( @Nonnull String path ) throws IOException |     public ReadableByteChannel openChannelForRead( @Nonnull String path ) throws IOException | ||||||
|     { |     { | ||||||
|         throw new IOException( "/" + path + ": No such file" ); |         throw new FileOperationException( path, "No such file" ); | ||||||
|     } |     } | ||||||
| } | } | ||||||
|   | |||||||
| @@ -7,6 +7,7 @@ | |||||||
| package dan200.computercraft.core.filesystem; | package dan200.computercraft.core.filesystem; | ||||||
|  |  | ||||||
| import com.google.common.collect.Sets; | import com.google.common.collect.Sets; | ||||||
|  | import dan200.computercraft.api.filesystem.FileOperationException; | ||||||
| import dan200.computercraft.api.filesystem.IWritableMount; | import dan200.computercraft.api.filesystem.IWritableMount; | ||||||
|  |  | ||||||
| import javax.annotation.Nonnull; | import javax.annotation.Nonnull; | ||||||
| @@ -167,12 +168,12 @@ public class FileMount implements IWritableMount | |||||||
|     { |     { | ||||||
|         if( !created() ) |         if( !created() ) | ||||||
|         { |         { | ||||||
|             if( !path.isEmpty() ) throw new IOException( "/" + path + ": Not a directory" ); |             if( !path.isEmpty() ) throw new FileOperationException( path, "Not a directory" ); | ||||||
|             return; |             return; | ||||||
|         } |         } | ||||||
|  |  | ||||||
|         File file = getRealPath( path ); |         File file = getRealPath( path ); | ||||||
|         if( !file.exists() || !file.isDirectory() ) throw new IOException( "/" + path + ": Not a directory" ); |         if( !file.exists() || !file.isDirectory() ) throw new FileOperationException( path, "Not a directory" ); | ||||||
|  |  | ||||||
|         String[] paths = file.list(); |         String[] paths = file.list(); | ||||||
|         for( String subPath : paths ) |         for( String subPath : paths ) | ||||||
| @@ -194,7 +195,7 @@ public class FileMount implements IWritableMount | |||||||
|             if( file.exists() ) return file.isDirectory() ? 0 : file.length(); |             if( file.exists() ) return file.isDirectory() ? 0 : file.length(); | ||||||
|         } |         } | ||||||
|  |  | ||||||
|         throw new IOException( "/" + path + ": No such file" ); |         throw new FileOperationException( path, "No such file" ); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @Nonnull |     @Nonnull | ||||||
| @@ -208,7 +209,7 @@ public class FileMount implements IWritableMount | |||||||
|             if( file.exists() && !file.isDirectory() ) return new FileInputStream( file ); |             if( file.exists() && !file.isDirectory() ) return new FileInputStream( file ); | ||||||
|         } |         } | ||||||
|  |  | ||||||
|         throw new IOException( "/" + path + ": No such file" ); |         throw new FileOperationException( path, "No such file" ); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @Nonnull |     @Nonnull | ||||||
| @@ -221,7 +222,7 @@ public class FileMount implements IWritableMount | |||||||
|             if( file.exists() && !file.isDirectory() ) return FileChannel.open( file.toPath(), READ_OPTIONS ); |             if( file.exists() && !file.isDirectory() ) return FileChannel.open( file.toPath(), READ_OPTIONS ); | ||||||
|         } |         } | ||||||
|  |  | ||||||
|         throw new IOException( "/" + path + ": No such file" ); |         throw new FileOperationException( path, "No such file" ); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     // IWritableMount implementation |     // IWritableMount implementation | ||||||
| @@ -233,7 +234,7 @@ public class FileMount implements IWritableMount | |||||||
|         File file = getRealPath( path ); |         File file = getRealPath( path ); | ||||||
|         if( file.exists() ) |         if( file.exists() ) | ||||||
|         { |         { | ||||||
|             if( !file.isDirectory() ) throw new IOException( "/" + path + ": File exists" ); |             if( !file.isDirectory() ) throw new FileOperationException( path, "File exists" ); | ||||||
|             return; |             return; | ||||||
|         } |         } | ||||||
|  |  | ||||||
| @@ -247,7 +248,7 @@ public class FileMount implements IWritableMount | |||||||
|  |  | ||||||
|         if( getRemainingSpace() < dirsToCreate * MINIMUM_FILE_SIZE ) |         if( getRemainingSpace() < dirsToCreate * MINIMUM_FILE_SIZE ) | ||||||
|         { |         { | ||||||
|             throw new IOException( "/" + path + ": Out of space" ); |             throw new FileOperationException( path, "Out of space" ); | ||||||
|         } |         } | ||||||
|  |  | ||||||
|         if( file.mkdirs() ) |         if( file.mkdirs() ) | ||||||
| @@ -256,14 +257,14 @@ public class FileMount implements IWritableMount | |||||||
|         } |         } | ||||||
|         else |         else | ||||||
|         { |         { | ||||||
|             throw new IOException( "/" + path + ": Access denied" ); |             throw new FileOperationException( path, "Access denied" ); | ||||||
|         } |         } | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @Override |     @Override | ||||||
|     public void delete( @Nonnull String path ) throws IOException |     public void delete( @Nonnull String path ) throws IOException | ||||||
|     { |     { | ||||||
|         if( path.isEmpty() ) throw new IOException( "/" + path + ": Access denied" ); |         if( path.isEmpty() ) throw new FileOperationException( path, "Access denied" ); | ||||||
|  |  | ||||||
|         if( created() ) |         if( created() ) | ||||||
|         { |         { | ||||||
| @@ -319,7 +320,7 @@ public class FileMount implements IWritableMount | |||||||
|     { |     { | ||||||
|         create(); |         create(); | ||||||
|         File file = getRealPath( path ); |         File file = getRealPath( path ); | ||||||
|         if( file.exists() && file.isDirectory() ) throw new IOException( "/" + path + ": Cannot write to directory" ); |         if( file.exists() && file.isDirectory() ) throw new FileOperationException( path, "Cannot write to directory" ); | ||||||
|  |  | ||||||
|         if( file.exists() ) |         if( file.exists() ) | ||||||
|         { |         { | ||||||
| @@ -327,7 +328,7 @@ public class FileMount implements IWritableMount | |||||||
|         } |         } | ||||||
|         else if( getRemainingSpace() < MINIMUM_FILE_SIZE ) |         else if( getRemainingSpace() < MINIMUM_FILE_SIZE ) | ||||||
|         { |         { | ||||||
|             throw new IOException( "/" + path + ": Out of space" ); |             throw new FileOperationException( path, "Out of space" ); | ||||||
|         } |         } | ||||||
|         m_usedSpace += MINIMUM_FILE_SIZE; |         m_usedSpace += MINIMUM_FILE_SIZE; | ||||||
|  |  | ||||||
| @@ -340,12 +341,12 @@ public class FileMount implements IWritableMount | |||||||
|     { |     { | ||||||
|         if( !created() ) |         if( !created() ) | ||||||
|         { |         { | ||||||
|             throw new IOException( "/" + path + ": No such file" ); |             throw new FileOperationException( path, "No such file" ); | ||||||
|         } |         } | ||||||
|  |  | ||||||
|         File file = getRealPath( path ); |         File file = getRealPath( path ); | ||||||
|         if( !file.exists() ) throw new IOException( "/" + path + ": No such file" ); |         if( !file.exists() ) throw new FileOperationException( path, "No such file" ); | ||||||
|         if( file.isDirectory() ) throw new IOException( "/" + path + ": Cannot write to directory" ); |         if( file.isDirectory() ) throw new FileOperationException( path, "Cannot write to directory" ); | ||||||
|  |  | ||||||
|         // Allowing seeking when appending is not recommended, so we use a separate channel. |         // Allowing seeking when appending is not recommended, so we use a separate channel. | ||||||
|         return new WritableCountingChannel( |         return new WritableCountingChannel( | ||||||
|   | |||||||
| @@ -8,6 +8,7 @@ package dan200.computercraft.core.filesystem; | |||||||
|  |  | ||||||
| import com.google.common.io.ByteStreams; | import com.google.common.io.ByteStreams; | ||||||
| import dan200.computercraft.ComputerCraft; | import dan200.computercraft.ComputerCraft; | ||||||
|  | import dan200.computercraft.api.filesystem.FileOperationException; | ||||||
| import dan200.computercraft.api.filesystem.IFileSystem; | import dan200.computercraft.api.filesystem.IFileSystem; | ||||||
| import dan200.computercraft.api.filesystem.IMount; | import dan200.computercraft.api.filesystem.IMount; | ||||||
| import dan200.computercraft.api.filesystem.IWritableMount; | import dan200.computercraft.api.filesystem.IWritableMount; | ||||||
| @@ -45,7 +46,7 @@ public class FileSystem | |||||||
|             m_writableMount = null; |             m_writableMount = null; | ||||||
|         } |         } | ||||||
|  |  | ||||||
|         public MountWrapper( String label, String location, IWritableMount mount ) |         MountWrapper( String label, String location, IWritableMount mount ) | ||||||
|         { |         { | ||||||
|             this( label, location, (IMount) mount ); |             this( label, location, (IMount) mount ); | ||||||
|             m_writableMount = mount; |             m_writableMount = mount; | ||||||
| @@ -107,7 +108,7 @@ public class FileSystem | |||||||
|             } |             } | ||||||
|             catch( IOException e ) |             catch( IOException e ) | ||||||
|             { |             { | ||||||
|                 throw new FileSystemException( e.getMessage() ); |                 throw localExceptionOf( e ); | ||||||
|             } |             } | ||||||
|         } |         } | ||||||
|  |  | ||||||
| @@ -122,12 +123,12 @@ public class FileSystem | |||||||
|                 } |                 } | ||||||
|                 else |                 else | ||||||
|                 { |                 { | ||||||
|                     throw new FileSystemException( "/" + path + ": Not a directory" ); |                     throw localExceptionOf( path, "Not a directory" ); | ||||||
|                 } |                 } | ||||||
|             } |             } | ||||||
|             catch( IOException e ) |             catch( IOException e ) | ||||||
|             { |             { | ||||||
|                 throw new FileSystemException( e.getMessage() ); |                 throw localExceptionOf( e ); | ||||||
|             } |             } | ||||||
|         } |         } | ||||||
|  |  | ||||||
| @@ -149,12 +150,12 @@ public class FileSystem | |||||||
|                 } |                 } | ||||||
|                 else |                 else | ||||||
|                 { |                 { | ||||||
|                     throw new FileSystemException( "/" + path + ": No such file" ); |                     throw localExceptionOf( path, "No such file" ); | ||||||
|                 } |                 } | ||||||
|             } |             } | ||||||
|             catch( IOException e ) |             catch( IOException e ) | ||||||
|             { |             { | ||||||
|                 throw new FileSystemException( e.getMessage() ); |                 throw localExceptionOf( e ); | ||||||
|             } |             } | ||||||
|         } |         } | ||||||
|  |  | ||||||
| @@ -169,12 +170,12 @@ public class FileSystem | |||||||
|                 } |                 } | ||||||
|                 else |                 else | ||||||
|                 { |                 { | ||||||
|                     throw new FileSystemException( "/" + path + ": No such file" ); |                     throw localExceptionOf( path, "No such file" ); | ||||||
|                 } |                 } | ||||||
|             } |             } | ||||||
|             catch( IOException e ) |             catch( IOException e ) | ||||||
|             { |             { | ||||||
|                 throw new FileSystemException( e.getMessage() ); |                 throw localExceptionOf( e ); | ||||||
|             } |             } | ||||||
|         } |         } | ||||||
|  |  | ||||||
| @@ -182,19 +183,14 @@ public class FileSystem | |||||||
|  |  | ||||||
|         public void makeDirectory( String path ) throws FileSystemException |         public void makeDirectory( String path ) throws FileSystemException | ||||||
|         { |         { | ||||||
|             if( m_writableMount == null ) |             if( m_writableMount == null ) throw exceptionOf( path, "Access denied" ); | ||||||
|             { |  | ||||||
|                 throw new FileSystemException( "/" + path + ": Access denied" ); |             path = toLocal( path ); | ||||||
|             } |  | ||||||
|             try |             try | ||||||
|             { |             { | ||||||
|                 path = toLocal( path ); |  | ||||||
|                 if( m_mount.exists( path ) ) |                 if( m_mount.exists( path ) ) | ||||||
|                 { |                 { | ||||||
|                     if( !m_mount.isDirectory( path ) ) |                     if( !m_mount.isDirectory( path ) ) throw localExceptionOf( path, "File exists" ); | ||||||
|                     { |  | ||||||
|                         throw new FileSystemException( "/" + path + ": File exists" ); |  | ||||||
|                     } |  | ||||||
|                 } |                 } | ||||||
|                 else |                 else | ||||||
|                 { |                 { | ||||||
| @@ -203,16 +199,14 @@ public class FileSystem | |||||||
|             } |             } | ||||||
|             catch( IOException e ) |             catch( IOException e ) | ||||||
|             { |             { | ||||||
|                 throw new FileSystemException( e.getMessage() ); |                 throw localExceptionOf( e ); | ||||||
|             } |             } | ||||||
|         } |         } | ||||||
|  |  | ||||||
|         public void delete( String path ) throws FileSystemException |         public void delete( String path ) throws FileSystemException | ||||||
|         { |         { | ||||||
|             if( m_writableMount == null ) |             if( m_writableMount == null ) throw exceptionOf( path, "Access denied" ); | ||||||
|             { |  | ||||||
|                 throw new FileSystemException( "/" + path + ": Access denied" ); |  | ||||||
|             } |  | ||||||
|             try |             try | ||||||
|             { |             { | ||||||
|                 path = toLocal( path ); |                 path = toLocal( path ); | ||||||
| @@ -227,22 +221,20 @@ public class FileSystem | |||||||
|             } |             } | ||||||
|             catch( IOException e ) |             catch( IOException e ) | ||||||
|             { |             { | ||||||
|                 throw new FileSystemException( e.getMessage() ); |                 throw localExceptionOf( e ); | ||||||
|             } |             } | ||||||
|         } |         } | ||||||
|  |  | ||||||
|         public WritableByteChannel openForWrite( String path ) throws FileSystemException |         public WritableByteChannel openForWrite( String path ) throws FileSystemException | ||||||
|         { |         { | ||||||
|             if( m_writableMount == null ) |             if( m_writableMount == null ) throw exceptionOf( path, "Access denied" ); | ||||||
|             { |  | ||||||
|                 throw new FileSystemException( "/" + path + ": Access denied" ); |             path = toLocal( path ); | ||||||
|             } |  | ||||||
|             try |             try | ||||||
|             { |             { | ||||||
|                 path = toLocal( path ); |  | ||||||
|                 if( m_mount.exists( path ) && m_mount.isDirectory( path ) ) |                 if( m_mount.exists( path ) && m_mount.isDirectory( path ) ) | ||||||
|                 { |                 { | ||||||
|                     throw new FileSystemException( "/" + path + ": Cannot write to directory" ); |                     throw localExceptionOf( path, "Cannot write to directory" ); | ||||||
|                 } |                 } | ||||||
|                 else |                 else | ||||||
|                 { |                 { | ||||||
| @@ -263,19 +255,17 @@ public class FileSystem | |||||||
|             } |             } | ||||||
|             catch( IOException e ) |             catch( IOException e ) | ||||||
|             { |             { | ||||||
|                 throw new FileSystemException( e.getMessage() ); |                 throw localExceptionOf( e ); | ||||||
|             } |             } | ||||||
|         } |         } | ||||||
|  |  | ||||||
|         public WritableByteChannel openForAppend( String path ) throws FileSystemException |         public WritableByteChannel openForAppend( String path ) throws FileSystemException | ||||||
|         { |         { | ||||||
|             if( m_writableMount == null ) |             if( m_writableMount == null ) throw exceptionOf( path, "Access denied" ); | ||||||
|             { |  | ||||||
|                 throw new FileSystemException( "/" + path + ": Access denied" ); |             path = toLocal( path ); | ||||||
|             } |  | ||||||
|             try |             try | ||||||
|             { |             { | ||||||
|                 path = toLocal( path ); |  | ||||||
|                 if( !m_mount.exists( path ) ) |                 if( !m_mount.exists( path ) ) | ||||||
|                 { |                 { | ||||||
|                     if( !path.isEmpty() ) |                     if( !path.isEmpty() ) | ||||||
| @@ -290,7 +280,7 @@ public class FileSystem | |||||||
|                 } |                 } | ||||||
|                 else if( m_mount.isDirectory( path ) ) |                 else if( m_mount.isDirectory( path ) ) | ||||||
|                 { |                 { | ||||||
|                     throw new FileSystemException( "/" + path + ": Cannot write to directory" ); |                     throw localExceptionOf( path, "Cannot write to directory" ); | ||||||
|                 } |                 } | ||||||
|                 else |                 else | ||||||
|                 { |                 { | ||||||
| @@ -303,16 +293,36 @@ public class FileSystem | |||||||
|             } |             } | ||||||
|             catch( IOException e ) |             catch( IOException e ) | ||||||
|             { |             { | ||||||
|                 throw new FileSystemException( e.getMessage() ); |                 throw localExceptionOf( e ); | ||||||
|             } |             } | ||||||
|         } |         } | ||||||
|  |  | ||||||
|         // private members |  | ||||||
|  |  | ||||||
|         private String toLocal( String path ) |         private String toLocal( String path ) | ||||||
|         { |         { | ||||||
|             return FileSystem.toLocal( path, m_location ); |             return FileSystem.toLocal( path, m_location ); | ||||||
|         } |         } | ||||||
|  |  | ||||||
|  |         private FileSystemException localExceptionOf( IOException e ) | ||||||
|  |         { | ||||||
|  |             if( !m_location.isEmpty() && e instanceof FileOperationException ) | ||||||
|  |             { | ||||||
|  |                 FileOperationException ex = (FileOperationException) e; | ||||||
|  |                 if( ex.getFilename() != null ) return localExceptionOf( ex.getFilename(), ex.getMessage() ); | ||||||
|  |             } | ||||||
|  |  | ||||||
|  |             return new FileSystemException( e.getMessage() ); | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         private FileSystemException localExceptionOf( String path, String message ) | ||||||
|  |         { | ||||||
|  |             if( !m_location.isEmpty() ) path = path.isEmpty() ? m_location : m_location + "/" + path; | ||||||
|  |             return exceptionOf( path, message ); | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         private static FileSystemException exceptionOf( String path, String message ) | ||||||
|  |         { | ||||||
|  |             return new FileSystemException( "/" + path + ": " + message ); | ||||||
|  |         } | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     private final FileSystemWrapperMount m_wrapper = new FileSystemWrapperMount( this ); |     private final FileSystemWrapperMount m_wrapper = new FileSystemWrapperMount( this ); | ||||||
| @@ -769,7 +779,7 @@ public class FileSystem | |||||||
|  |  | ||||||
|         // Clean the path or illegal characters. |         // Clean the path or illegal characters. | ||||||
|         final char[] specialChars = new char[] { |         final char[] specialChars = new char[] { | ||||||
|             '"', ':', '<', '>', '?', '|' // Sorted by ascii value (important) |             '"', ':', '<', '>', '?', '|', // Sorted by ascii value (important) | ||||||
|         }; |         }; | ||||||
|  |  | ||||||
|         StringBuilder cleanName = new StringBuilder(); |         StringBuilder cleanName = new StringBuilder(); | ||||||
|   | |||||||
| @@ -9,6 +9,7 @@ package dan200.computercraft.core.filesystem; | |||||||
| import com.google.common.cache.Cache; | import com.google.common.cache.Cache; | ||||||
| import com.google.common.cache.CacheBuilder; | import com.google.common.cache.CacheBuilder; | ||||||
| import com.google.common.io.ByteStreams; | import com.google.common.io.ByteStreams; | ||||||
|  | import dan200.computercraft.api.filesystem.FileOperationException; | ||||||
| import dan200.computercraft.api.filesystem.IMount; | import dan200.computercraft.api.filesystem.IMount; | ||||||
| import dan200.computercraft.core.apis.handles.ArrayByteChannel; | import dan200.computercraft.core.apis.handles.ArrayByteChannel; | ||||||
| import dan200.computercraft.shared.util.IoUtil; | import dan200.computercraft.shared.util.IoUtil; | ||||||
| @@ -92,7 +93,7 @@ public class JarMount implements IMount | |||||||
|         new MountReference( this ); |         new MountReference( this ); | ||||||
|  |  | ||||||
|         // Read in all the entries |         // Read in all the entries | ||||||
|         root = new FileEntry( "" ); |         root = new FileEntry(); | ||||||
|         Enumeration<? extends ZipEntry> zipEntries = zip.entries(); |         Enumeration<? extends ZipEntry> zipEntries = zip.entries(); | ||||||
|         while( zipEntries.hasMoreElements() ) |         while( zipEntries.hasMoreElements() ) | ||||||
|         { |         { | ||||||
| @@ -139,7 +140,7 @@ public class JarMount implements IMount | |||||||
|             FileEntry nextEntry = lastEntry.children.get( part ); |             FileEntry nextEntry = lastEntry.children.get( part ); | ||||||
|             if( nextEntry == null || !nextEntry.isDirectory() ) |             if( nextEntry == null || !nextEntry.isDirectory() ) | ||||||
|             { |             { | ||||||
|                 lastEntry.children.put( part, nextEntry = new FileEntry( part ) ); |                 lastEntry.children.put( part, nextEntry = new FileEntry() ); | ||||||
|             } |             } | ||||||
|  |  | ||||||
|             lastEntry = nextEntry; |             lastEntry = nextEntry; | ||||||
| @@ -166,7 +167,7 @@ public class JarMount implements IMount | |||||||
|     public void list( @Nonnull String path, @Nonnull List<String> contents ) throws IOException |     public void list( @Nonnull String path, @Nonnull List<String> contents ) throws IOException | ||||||
|     { |     { | ||||||
|         FileEntry file = get( path ); |         FileEntry file = get( path ); | ||||||
|         if( file == null || !file.isDirectory() ) throw new IOException( "/" + path + ": Not a directory" ); |         if( file == null || !file.isDirectory() ) throw new FileOperationException( path, "Not a directory" ); | ||||||
|  |  | ||||||
|         file.list( contents ); |         file.list( contents ); | ||||||
|     } |     } | ||||||
| @@ -176,7 +177,7 @@ public class JarMount implements IMount | |||||||
|     { |     { | ||||||
|         FileEntry file = get( path ); |         FileEntry file = get( path ); | ||||||
|         if( file != null ) return file.size; |         if( file != null ) return file.size; | ||||||
|         throw new IOException( "/" + path + ": No such file" ); |         throw new FileOperationException( path, "No such file" ); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     @Nonnull |     @Nonnull | ||||||
| @@ -218,22 +219,15 @@ public class JarMount implements IMount | |||||||
|             } |             } | ||||||
|         } |         } | ||||||
|  |  | ||||||
|         throw new IOException( "/" + path + ": No such file" ); |         throw new FileOperationException( path, "No such file" ); | ||||||
|     } |     } | ||||||
|  |  | ||||||
|     private static class FileEntry |     private static class FileEntry | ||||||
|     { |     { | ||||||
|         final String name; |  | ||||||
|  |  | ||||||
|         String path; |         String path; | ||||||
|         long size; |         long size; | ||||||
|         Map<String, FileEntry> children; |         Map<String, FileEntry> children; | ||||||
|  |  | ||||||
|         FileEntry( String name ) |  | ||||||
|         { |  | ||||||
|             this.name = name; |  | ||||||
|         } |  | ||||||
|  |  | ||||||
|         void setup( ZipEntry entry ) |         void setup( ZipEntry entry ) | ||||||
|         { |         { | ||||||
|             path = entry.getName(); |             path = entry.getName(); | ||||||
|   | |||||||
| @@ -8,13 +8,16 @@ package dan200.computercraft.core.filesystem; | |||||||
|  |  | ||||||
| import com.google.common.cache.Cache; | import com.google.common.cache.Cache; | ||||||
| import com.google.common.cache.CacheBuilder; | import com.google.common.cache.CacheBuilder; | ||||||
|  | import com.google.common.collect.MapMaker; | ||||||
| import com.google.common.io.ByteStreams; | import com.google.common.io.ByteStreams; | ||||||
|  | import dan200.computercraft.ComputerCraft; | ||||||
| import dan200.computercraft.api.filesystem.IMount; | import dan200.computercraft.api.filesystem.IMount; | ||||||
| import dan200.computercraft.core.apis.handles.ArrayByteChannel; | import dan200.computercraft.core.apis.handles.ArrayByteChannel; | ||||||
| import net.minecraft.resources.IReloadableResourceManager; | import net.minecraft.resources.IReloadableResourceManager; | ||||||
| import net.minecraft.resources.IResource; | import net.minecraft.resources.IResource; | ||||||
| import net.minecraft.resources.IResourceManager; | import net.minecraft.resources.IResourceManager; | ||||||
| import net.minecraft.util.ResourceLocation; | import net.minecraft.util.ResourceLocation; | ||||||
|  | import net.minecraft.util.ResourceLocationException; | ||||||
| import net.minecraftforge.resource.IResourceType; | import net.minecraftforge.resource.IResourceType; | ||||||
| import net.minecraftforge.resource.ISelectiveResourceReloadListener; | import net.minecraftforge.resource.ISelectiveResourceReloadListener; | ||||||
|  |  | ||||||
| @@ -29,7 +32,7 @@ import java.util.*; | |||||||
| import java.util.concurrent.TimeUnit; | import java.util.concurrent.TimeUnit; | ||||||
| import java.util.function.Predicate; | import java.util.function.Predicate; | ||||||
|  |  | ||||||
| public class ResourceMount implements IMount | public final class ResourceMount implements IMount | ||||||
| { | { | ||||||
|     /** |     /** | ||||||
|      * Only cache files smaller than 1MiB. |      * Only cache files smaller than 1MiB. | ||||||
| @@ -55,6 +58,13 @@ public class ResourceMount implements IMount | |||||||
|         .<FileEntry, byte[]>weigher( ( k, v ) -> v.length ) |         .<FileEntry, byte[]>weigher( ( k, v ) -> v.length ) | ||||||
|         .build(); |         .build(); | ||||||
|  |  | ||||||
|  |     private static final MapMaker CACHE_TEMPLATE = new MapMaker().weakValues().concurrencyLevel( 1 ); | ||||||
|  |  | ||||||
|  |     /** | ||||||
|  |      * Maintain a cache of currently loaded resource mounts. This cache is invalidated when currentManager changes. | ||||||
|  |      */ | ||||||
|  |     private static final Map<IReloadableResourceManager, Map<ResourceLocation, ResourceMount>> MOUNT_CACHE = new WeakHashMap<>( 2 ); | ||||||
|  |  | ||||||
|     private final String namespace; |     private final String namespace; | ||||||
|     private final String subPath; |     private final String subPath; | ||||||
|     private final IReloadableResourceManager manager; |     private final IReloadableResourceManager manager; | ||||||
| @@ -62,7 +72,26 @@ public class ResourceMount implements IMount | |||||||
|     @Nullable |     @Nullable | ||||||
|     private FileEntry root; |     private FileEntry root; | ||||||
|  |  | ||||||
|     public ResourceMount( String namespace, String subPath, IReloadableResourceManager manager ) |     public static ResourceMount get( String namespace, String subPath, IReloadableResourceManager manager ) | ||||||
|  |     { | ||||||
|  |         Map<ResourceLocation, ResourceMount> cache; | ||||||
|  |  | ||||||
|  |         synchronized( MOUNT_CACHE ) | ||||||
|  |         { | ||||||
|  |             cache = MOUNT_CACHE.get( manager ); | ||||||
|  |             if( cache == null ) MOUNT_CACHE.put( manager, cache = CACHE_TEMPLATE.makeMap() ); | ||||||
|  |         } | ||||||
|  |  | ||||||
|  |         ResourceLocation path = new ResourceLocation( namespace, subPath ); | ||||||
|  |         synchronized( cache ) | ||||||
|  |         { | ||||||
|  |             ResourceMount mount = cache.get( path ); | ||||||
|  |             if( mount == null ) cache.put( path, mount = new ResourceMount( namespace, subPath, manager ) ); | ||||||
|  |             return mount; | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  |  | ||||||
|  |     private ResourceMount( String namespace, String subPath, IReloadableResourceManager manager ) | ||||||
|     { |     { | ||||||
|         this.namespace = namespace; |         this.namespace = namespace; | ||||||
|         this.subPath = subPath; |         this.subPath = subPath; | ||||||
| @@ -119,7 +148,17 @@ public class ResourceMount implements IMount | |||||||
|             FileEntry nextEntry = lastEntry.children.get( part ); |             FileEntry nextEntry = lastEntry.children.get( part ); | ||||||
|             if( nextEntry == null ) |             if( nextEntry == null ) | ||||||
|             { |             { | ||||||
|                 lastEntry.children.put( part, nextEntry = new FileEntry( new ResourceLocation( namespace, subPath + "/" + path ) ) ); |                 ResourceLocation childPath; | ||||||
|  |                 try | ||||||
|  |                 { | ||||||
|  |                     childPath = new ResourceLocation( namespace, subPath + "/" + path ); | ||||||
|  |                 } | ||||||
|  |                 catch( ResourceLocationException e ) | ||||||
|  |                 { | ||||||
|  |                     ComputerCraft.log.warn( "Cannot create resource location for {} ({})", part, e.getMessage() ); | ||||||
|  |                     return; | ||||||
|  |                 } | ||||||
|  |                 lastEntry.children.put( part, nextEntry = new FileEntry( childPath ) ); | ||||||
|             } |             } | ||||||
|  |  | ||||||
|             lastEntry = nextEntry; |             lastEntry = nextEntry; | ||||||
|   | |||||||
| @@ -550,7 +550,7 @@ public class CobaltLuaMachine implements ILuaMachine | |||||||
|                 { |                 { | ||||||
|                     if( ComputerCraft.logPeripheralErrors ) ComputerCraft.log.error( "Error running task", t ); |                     if( ComputerCraft.logPeripheralErrors ) ComputerCraft.log.error( "Error running task", t ); | ||||||
|                     m_computer.queueEvent( "task_complete", new Object[] { |                     m_computer.queueEvent( "task_complete", new Object[] { | ||||||
|                         taskID, false, "Java Exception Thrown: " + t |                         taskID, false, "Java Exception Thrown: " + t, | ||||||
|                     } ); |                     } ); | ||||||
|                 } |                 } | ||||||
|             }; |             }; | ||||||
|   | |||||||
| @@ -38,7 +38,7 @@ public final class MachineResult | |||||||
|     public static final MachineResult TIMEOUT = new MachineResult( true, false, TimeoutState.ABORT_MESSAGE ); |     public static final MachineResult TIMEOUT = new MachineResult( true, false, TimeoutState.ABORT_MESSAGE ); | ||||||
|  |  | ||||||
|     /** |     /** | ||||||
|      * An error with no user-friendly error message |      * An error with no user-friendly error message. | ||||||
|      */ |      */ | ||||||
|     public static final MachineResult GENERIC_ERROR = new MachineResult( true, false, null ); |     public static final MachineResult GENERIC_ERROR = new MachineResult( true, false, null ); | ||||||
|  |  | ||||||
|   | |||||||
Some files were not shown because too many files have changed in this diff Show More
		Reference in New Issue
	
	Block a user