Compare commits
	
		
			141 Commits
		
	
	
		
			v1.16.5-1.
			...
			v1.92.0
		
	
	| Author | SHA1 | Date | |
|---|---|---|---|
|   | 19273b3696 | ||
|   | 4643641d51 | ||
|   | 63cd9c5bc7 | ||
|   | e9c11ff325 | ||
|   | 452464aa01 | ||
|   | 8885462175 | ||
|   | b8bd64913b | ||
|   | 88722d484f | ||
|   | 6d103e2114 | ||
|   | 42f23d56ae | ||
|   | 89d5211bd7 | ||
|   | 83e70377f7 | ||
|   | f6a26f75c3 | ||
|   | 664df62d5d | ||
|   | 1348ee0588 | ||
|   | 145dce7653 | ||
|   | 7f2651c23e | ||
|   | 05464107a8 | ||
|   | 86705787f0 | ||
|   | b34d8387d9 | ||
|   | 4d00969ef0 | ||
|   | 01d3d12992 | ||
|   | 5e31dcde83 | ||
|   | 5184883af1 | ||
|   | 0c45112262 | ||
|   | 0bf1672f45 | ||
|   | e1b8ac1f84 | ||
|   | deea552d99 | ||
|   | 54229c2ce1 | ||
|   | 1346a26179 | ||
|   | 14df44f09d | ||
|   | 1a21529499 | ||
|   | 2546990f41 | ||
|   | b39ca02464 | ||
|   | 628618105c | ||
|   | 075ba03f5d | ||
|   | 10bf84b631 | ||
|   | c346e22a45 | ||
|   | 418e1335b1 | ||
|   | d52372df31 | ||
|   | cc72e1c2bd | ||
|   | d28afcc6a9 | ||
|   | 220ed21e6e | ||
|   | 09d465774d | ||
|   | dfc8f48f12 | ||
|   | abe2ec4686 | ||
|   | 5333cda44e | ||
|   | dc88fbeb12 | ||
|   | c5eb7a9501 | ||
|   | bc969db2be | ||
|   | 50b7646178 | ||
|   | 3fa6b5bc9d | ||
|   | 33e65e39e3 | ||
|   | 0380e60590 | ||
|   | 2918892ee2 | ||
|   | e43dd9f7c6 | ||
|   | 244fd95034 | ||
|   | 14e98e2fcb | ||
|   | dcf5d59109 | ||
|   | b95083c77e | ||
|   | c54c8c3ea6 | ||
|   | f79c67e243 | ||
|   | 6ec7ebe439 | ||
|   | 76fe33760d | ||
|   | cb549d8f43 | ||
|   | 7b400fdcdd | ||
|   | ab70d918b5 | ||
|   | c64644b9ec | ||
|   | afb12eb342 | ||
|   | 64f5ca02b3 | ||
|   | 84bca21b0c | ||
|   | b6757c416f | ||
|   | dc9d3f2d15 | ||
|   | 807825d74e | ||
|   | 14c17676c6 | ||
|   | 324519575c | ||
|   | fbcf26bdc9 | ||
|   | 30ab6bd045 | ||
|   | 2ea816b78b | ||
|   | 56dcc57755 | ||
|   | a4830aff86 | ||
|   | 621bc526be | ||
|   | 605e1f6b9b | ||
|   | cb66ef7e30 | ||
|   | fd2f6a38c1 | ||
|   | 229821d398 | ||
|   | 1f2e0c444d | ||
|   | 429baa350c | ||
|   | 7fe62485fa | ||
|   | fe10c68099 | ||
|   | 06092cfddd | ||
|   | 9967dc5740 | ||
|   | b920e04c59 | ||
|   | baa1b5a5c9 | ||
|   | e84663a5c5 | ||
|   | 1c46220c42 | ||
|   | 889b445855 | ||
|   | 19553a981e | ||
|   | 15f4dbd061 | ||
|   | 98e12c7c3e | ||
|   | dfad319864 | ||
|   | 111b58f533 | ||
|   | 9345652808 | ||
|   | d40163b409 | ||
|   | 96400966d7 | ||
|   | db83bd4f64 | ||
|   | 579a38d366 | ||
|   | 0e6797c7da | ||
|   | c935577768 | ||
|   | c2b4077aa1 | ||
|   | ac020859f3 | ||
|   | 238be8955b | ||
|   | f997b02b8f | ||
|   | 799bb77847 | ||
|   | 2f66792a0d | ||
|   | 6cae8e211e | ||
|   | 48b1b2f01d | ||
|   | 8c45fd362a | ||
|   | 07b13dd2b4 | ||
|   | 45e84e1ede | ||
|   | 725dfa764f | ||
|   | c221502ec9 | ||
|   | 234f18c228 | ||
|   | 006ad109cb | ||
|   | 0db080154c | ||
|   | 200311033f | ||
|   | 3192dc81ac | ||
|   | b11d4bb209 | ||
|   | 2a716244e9 | ||
|   | 19b7ed538a | ||
|   | b0d9dc0b88 | ||
|   | e6094a59fa | ||
|   | e8d7e6a562 | ||
|   | 536c2d9b2d | ||
|   | f15a278f3b | ||
|   | 26b73c2ff3 | ||
|   | c1e08fc3c7 | ||
|   | b9ec6f236d | ||
|   | b1fff97bff | ||
|   | c81bc70475 | ||
|   | 55a7ee4acf | 
| @@ -11,8 +11,5 @@ insert_final_newline = true | ||||
| [*.md] | ||||
| trim_trailing_whitespace = false | ||||
|  | ||||
| [*.sexp] | ||||
| indent_size = 2 | ||||
|  | ||||
| [*.yml] | ||||
| indent_size = 2 | ||||
| [*.properties] | ||||
| insert_final_newline = false | ||||
|   | ||||
							
								
								
									
										16
									
								
								.gitattributes
									
									
									
									
										vendored
									
									
								
							
							
						
						| @@ -1,16 +0,0 @@ | ||||
| # Ignore changes in generated files | ||||
| src/generated/** linguist-generated | ||||
| src/testMod/server-files/structures linguist-generated | ||||
|  | ||||
| * text=auto | ||||
|  | ||||
| *.gradle eol=lf diff=java | ||||
| *.java   eol=lf diff=java | ||||
| *.kt     eol=lf diff=java | ||||
| *.lua    eol=lf | ||||
| *.md     eol=lf diff=markdown | ||||
| *.txt    eol=lf | ||||
|  | ||||
| *.png binary | ||||
| *.jar binary | ||||
| *.dfpwm binary | ||||
							
								
								
									
										15
									
								
								.github/ISSUE_TEMPLATE/bug_report.md
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,15 @@ | ||||
| --- | ||||
| name: Bug report | ||||
| about: Report some misbehaviour in the mod | ||||
| labels: bug | ||||
| --- | ||||
|  | ||||
| <!-- | ||||
| ## Before reporting | ||||
|  - Search for the bug on the issue tracker. Make sure to look at closed issues too! | ||||
| --> | ||||
|  | ||||
| ## Useful information to include: | ||||
|  - Minecraft version | ||||
|  - CC: Tweaked version | ||||
|  - Detailed reproduction steps: sometimes I can spot a bug pretty easily, but often it's much more obscure. The more information I have to help reproduce it, the quicker it'll get fixed. | ||||
							
								
								
									
										34
									
								
								.github/ISSUE_TEMPLATE/bug_report.yaml
									
									
									
									
										vendored
									
									
								
							
							
						
						| @@ -1,34 +0,0 @@ | ||||
| name: Bug report | ||||
| description: Report some misbehaviour in the mod | ||||
| labels: [ bug ] | ||||
| body: | ||||
| - type: dropdown | ||||
|   id: mc-version | ||||
|   attributes: | ||||
|     label: Minecraft Version | ||||
|     description: What version of Minecraft are you using? | ||||
|     options: | ||||
|       - 1.16.x | ||||
|       - 1.17.x | ||||
|       - 1.18.x | ||||
|   validations: | ||||
|     required: true | ||||
| - type: input | ||||
|   id: version | ||||
|   attributes: | ||||
|     label: Version | ||||
|     description: "What version of CC: Tweaked are you using?" | ||||
|     placeholder: "e.g. 1.96.0" | ||||
|   validations: | ||||
|     required: true | ||||
| - type: textarea | ||||
|   id: details | ||||
|   attributes: | ||||
|     label: Details | ||||
|     description: | | ||||
|       Description of the bug. Please include the following: | ||||
|       - Logs: These will be located in the `logs/` directory of your Minecraft | ||||
|         instance. Please upload them as a gist or directly into this editor. | ||||
|       - Detailed reproduction steps: sometimes I can spot a bug pretty easily, | ||||
|         but often it's much more obscure. The more information I have to help | ||||
|         reproduce it, the quicker it'll get fixed. | ||||
							
								
								
									
										8
									
								
								.github/ISSUE_TEMPLATE/config.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						| @@ -1,8 +0,0 @@ | ||||
| blank_issues_enabled: false | ||||
| contact_links: | ||||
| - name: ComputerCraft Discord | ||||
|   url: https://discord.computercraft.cc | ||||
|   about: Get help on the ComputerCraft Discord. | ||||
| - name: GitHub Discussions | ||||
|   url: https://github.com/cc-tweaked/CC-Tweaked/discussions | ||||
|   about: Or ask questions on GitHub Discussions. | ||||
							
								
								
									
										4
									
								
								.github/ISSUE_TEMPLATE/something_else.md
									
									
									
									
										vendored
									
									
								
							
							
						
						| @@ -1,4 +0,0 @@ | ||||
| --- | ||||
| name: Something else | ||||
| about: An issue about something else. | ||||
| --- | ||||
							
								
								
									
										17
									
								
								.github/matchers/checkstyle.json
									
									
									
									
										vendored
									
									
								
							
							
						
						| @@ -1,17 +0,0 @@ | ||||
| { | ||||
|     "problemMatcher": [ | ||||
|         { | ||||
|             "owner": "checkstyle", | ||||
|             "pattern": [ | ||||
|                 { | ||||
|                     "regexp": "^([a-z]+) ([\\w./-]+):(\\d+):(\\d+): (.*)$", | ||||
|                     "severity": 1, | ||||
|                     "file": 2, | ||||
|                     "line": 3, | ||||
|                     "column": 4, | ||||
|                     "message": 5 | ||||
|                 } | ||||
|             ] | ||||
|         } | ||||
|     ] | ||||
| } | ||||
							
								
								
									
										18
									
								
								.github/matchers/illuaminate.json
									
									
									
									
										vendored
									
									
								
							
							
						
						| @@ -1,18 +0,0 @@ | ||||
| { | ||||
|     "problemMatcher": [ | ||||
|         { | ||||
|             "owner": "illuaminate", | ||||
|             "severity": "warning", | ||||
|             "pattern": [ | ||||
|                 { | ||||
|                     "regexp": "^([\\w./-]+):\\[(\\d+):(\\d+)\\-(?:\\d+):(?:\\d+)\\]: (.*) \\[([a-z:-]+)\\]$", | ||||
|                     "file": 1, | ||||
|                     "line": 2, | ||||
|                     "column": 3, | ||||
|                     "message": 4, | ||||
|                     "code": 5 | ||||
|                 } | ||||
|             ] | ||||
|         } | ||||
|     ] | ||||
| } | ||||
							
								
								
									
										15
									
								
								.github/matchers/junit.json
									
									
									
									
										vendored
									
									
								
							
							
						
						| @@ -1,15 +0,0 @@ | ||||
| { | ||||
|     "problemMatcher": [ | ||||
|         { | ||||
|             "owner": "junit", | ||||
|             "pattern": [ | ||||
|                 { | ||||
|                     "regexp": "^## ([\\w./-]+):(\\d+): (.*)$", | ||||
|                     "file": 1, | ||||
|                     "line": 2, | ||||
|                     "message": 3 | ||||
|                 } | ||||
|             ] | ||||
|         } | ||||
|     ] | ||||
| } | ||||
							
								
								
									
										70
									
								
								.github/workflows/main-ci.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						| @@ -1,70 +0,0 @@ | ||||
| name: Build | ||||
|  | ||||
| on: [push, pull_request] | ||||
|  | ||||
| jobs: | ||||
|   build: | ||||
|     name: Build | ||||
|     runs-on: ubuntu-latest | ||||
|  | ||||
|     steps: | ||||
|     - uses: actions/checkout@v2 | ||||
|  | ||||
|     - name: Set up Java 8 | ||||
|       uses: actions/setup-java@v1 | ||||
|       with: | ||||
|         java-version: 8 | ||||
|  | ||||
|     - name: Cache gradle dependencies | ||||
|       uses: actions/cache@v2 | ||||
|       with: | ||||
|         path: ~/.gradle/caches | ||||
|         key: ${{ runner.os }}-gradle-${{ hashFiles('gradle.properties') }} | ||||
|         restore-keys: | | ||||
|           ${{ runner.os }}-gradle- | ||||
|  | ||||
|     - name: Disable Gradle daemon | ||||
|       run: | | ||||
|         mkdir -p ~/.gradle | ||||
|         echo "org.gradle.daemon=false" >> ~/.gradle/gradle.properties | ||||
|  | ||||
|     - name: Build with Gradle | ||||
|       run: | | ||||
|         ./gradlew assemble || ./gradlew assemble | ||||
|         ./gradlew downloadAssets || ./gradlew downloadAssets | ||||
|         xvfb-run ./gradlew build | ||||
|  | ||||
|     - name: Upload Jar | ||||
|       uses: actions/upload-artifact@v2 | ||||
|       with: | ||||
|         name: CC-Tweaked | ||||
|         path: build/libs | ||||
|  | ||||
|     - name: Upload Screnshots | ||||
|       uses: actions/upload-artifact@v2 | ||||
|       with: | ||||
|         name: Screenshots | ||||
|         path: test-files/client/screenshots | ||||
|         if-no-files-found: ignore | ||||
|         retention-days: 5 | ||||
|       if: failure() | ||||
|  | ||||
|     - name: Upload Coverage | ||||
|       uses: codecov/codecov-action@v2 | ||||
|  | ||||
|     - name: Parse test reports | ||||
|       run: ./tools/parse-reports.py | ||||
|       if: ${{ failure() }} | ||||
|  | ||||
|     - name: Cache pre-commit | ||||
|       uses: actions/cache@v2 | ||||
|       with: | ||||
|         path: ~/.cache/pre-commit | ||||
|         key: ${{ runner.os }}-pre-commit-${{ hashFiles('config/pre-commit/config.yml') }} | ||||
|         restore-keys: | | ||||
|           ${{ runner.os }}-pre-commit- | ||||
|  | ||||
|     - name: Run linters | ||||
|       run: | | ||||
|         pip install pre-commit | ||||
|         pre-commit run --config config/pre-commit/config.yml --show-diff-on-failure --all --color=always | ||||
							
								
								
									
										19
									
								
								.github/workflows/make-doc.sh
									
									
									
									
										vendored
									
									
								
							
							
						
						| @@ -1,19 +0,0 @@ | ||||
| #!/usr/bin/env bash | ||||
|  | ||||
| set -eu | ||||
|  | ||||
| DEST="${GITHUB_REF#refs/*/}" | ||||
| echo "Uploading docs to https://tweaked.cc/$DEST" | ||||
|  | ||||
| # Setup ssh key | ||||
| mkdir -p "$HOME/.ssh/" | ||||
| echo "$SSH_KEY" > "$HOME/.ssh/key" | ||||
| chmod 600 "$HOME/.ssh/key" | ||||
|  | ||||
| # And upload | ||||
| rsync -avc -e "ssh -i $HOME/.ssh/key -o StrictHostKeyChecking=no -p $SSH_PORT" \ | ||||
|       "$GITHUB_WORKSPACE/build/docs/site/" \ | ||||
|       "$SSH_USER@$SSH_HOST:/$DEST" | ||||
| rsync -avc -e "ssh -i $HOME/.ssh/key -o StrictHostKeyChecking=no -p $SSH_PORT" \ | ||||
|       "$GITHUB_WORKSPACE/build/docs/javadoc/" \ | ||||
|       "$SSH_USER@$SSH_HOST:/$DEST/javadoc" | ||||
							
								
								
									
										50
									
								
								.github/workflows/make-doc.yml
									
									
									
									
										vendored
									
									
								
							
							
						
						| @@ -1,50 +0,0 @@ | ||||
| name: Build documentation | ||||
|  | ||||
| on: | ||||
|   push: | ||||
|     branches: | ||||
|     - mc-1.16.x | ||||
|  | ||||
| jobs: | ||||
|   make_doc: | ||||
|     name: Build | ||||
|     runs-on: ubuntu-latest | ||||
|  | ||||
|     steps: | ||||
|     - uses: actions/checkout@v2 | ||||
|  | ||||
|     - name: Set up Java 8 | ||||
|       uses: actions/setup-java@v1 | ||||
|       with: | ||||
|         java-version: 8 | ||||
|  | ||||
|     - name: Cache gradle dependencies | ||||
|       uses: actions/cache@v2 | ||||
|       with: | ||||
|         path: ~/.gradle/caches | ||||
|         key: ${{ runner.os }}-gradle-${{ hashFiles('gradle.properties') }} | ||||
|         restore-keys: | | ||||
|           ${{ runner.os }}-gradle- | ||||
|  | ||||
|     - name: Setup illuaminate | ||||
|       run: | | ||||
|         test -d bin || mkdir bin | ||||
|         test -f bin/illuaminate || wget -q -Obin/illuaminate https://squiddev.cc/illuaminate/linux-x86-64/illuaminate | ||||
|         chmod +x bin/illuaminate | ||||
|  | ||||
|     - name: Setup node | ||||
|       run: npm ci | ||||
|  | ||||
|     - name: Build with Gradle | ||||
|       run: ./gradlew compileJava --no-daemon || ./gradlew compileJava --no-daemon | ||||
|  | ||||
|     - name: Generate documentation | ||||
|       run: ./gradlew docWebsite javadoc --no-daemon | ||||
|  | ||||
|     - name: Upload documentation | ||||
|       run: .github/workflows/make-doc.sh 2> /dev/null | ||||
|       env: | ||||
|         SSH_KEY:  ${{ secrets.SSH_KEY  }} | ||||
|         SSH_USER: ${{ secrets.SSH_USER }} | ||||
|         SSH_HOST: ${{ secrets.SSH_HOST }} | ||||
|         SSH_PORT: ${{ secrets.SSH_PORT }} | ||||
							
								
								
									
										13
									
								
								.gitignore
									
									
									
									
										vendored
									
									
								
							
							
						
						| @@ -3,8 +3,6 @@ | ||||
| /logs | ||||
| /build | ||||
| /out | ||||
| /doc/out/ | ||||
| /node_modules | ||||
|  | ||||
| # Runtime directories | ||||
| /run | ||||
| @@ -17,13 +15,4 @@ | ||||
| .idea | ||||
| .gradle | ||||
| *.DS_Store | ||||
|  | ||||
| /.classpath | ||||
| /.project | ||||
| /.settings | ||||
| /.vscode | ||||
| bin/ | ||||
| *.launch | ||||
|  | ||||
| /src/generated/resources/.cache | ||||
| /src/web/mount/*.d.ts | ||||
| .project | ||||
							
								
								
									
										22
									
								
								.gitpod.yml
									
									
									
									
									
								
							
							
						
						| @@ -1,22 +0,0 @@ | ||||
| image: | ||||
|   file: config/gitpod/Dockerfile | ||||
|  | ||||
| ports: | ||||
|   - port: 25565 | ||||
|     onOpen: notify | ||||
|  | ||||
| vscode: | ||||
|   extensions: | ||||
|     - eamodio.gitlens | ||||
|     - github.vscode-pull-request-github | ||||
|     - ms-azuretools.vscode-docker | ||||
|     - redhat.java | ||||
|     - richardwillis.vscode-gradle | ||||
|     - vscjava.vscode-java-debug | ||||
|     - vscode.github | ||||
|  | ||||
| tasks: | ||||
|   - name: Setup pre-commit hool | ||||
|     init: pre-commit install --config config/pre-commit/config.yml --allow-missing-config | ||||
|   - name: Install npm packages | ||||
|     init: npm ci | ||||
							
								
								
									
										34
									
								
								.luacheckrc
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,34 @@ | ||||
| std = "max" | ||||
|  | ||||
| ignore = { | ||||
|     -- Allow access to undefined globals or their fields. In the future we'll | ||||
|     -- define all of CC's globals within this file | ||||
|     '113', '143', | ||||
|  | ||||
|     -- FIXME: Ignore unused arguments and loop variables | ||||
|     '212', '213', | ||||
|  | ||||
|     -- Disable line is too long for now. It would be possible to clean | ||||
|     -- this up in the future. | ||||
|     '631', | ||||
| } | ||||
|  | ||||
| -- Only run the linter on ROM and bios for now, as the treasure disks | ||||
| -- are largely unsupported. | ||||
| include_files = { | ||||
|     'src/main/resources/assets/computercraft/lua/rom', | ||||
|     'src/main/resources/assets/computercraft/lua/bios.lua' | ||||
| } | ||||
|  | ||||
| files['src/main/resources/assets/computercraft/lua/bios.lua'] = { | ||||
|     -- Allow declaring and mutating globals | ||||
|     allow_defined_top = true, | ||||
|     ignore = { '112', '121', '122', '131', '142' }, | ||||
| } | ||||
|  | ||||
| files['src/main/resources/assets/computercraft/lua/rom/apis'] = { | ||||
|     -- APIs may define globals on the top level. We'll ignore unused globals, | ||||
|     -- as obviously they may be used outside that API. | ||||
|     allow_defined_top = true, | ||||
|     ignore = { '131' }, | ||||
| } | ||||
							
								
								
									
										14
									
								
								.travis.yml
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,14 @@ | ||||
| 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 | ||||
							
								
								
									
										3
									
								
								.vscode/settings.json
									
									
									
									
										vendored
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,3 @@ | ||||
| { | ||||
| 	"java.configuration.updateBuildConfiguration": "automatic" | ||||
| } | ||||
							
								
								
									
										115
									
								
								CONTRIBUTING.md
									
									
									
									
									
								
							
							
						
						| @@ -1,115 +0,0 @@ | ||||
| # Contributing to CC: Tweaked | ||||
| As with many open source projects, CC: Tweaked thrives on contributions from other people! This document (hopefully) | ||||
| provides an introduction as to how to get started in helping out. | ||||
|  | ||||
| If you've any other questions, [just ask the community][community] or [open an issue][new-issue]. | ||||
|  | ||||
| ## Reporting issues | ||||
| If you have a bug, suggestion, or other feedback, the best thing to do is [file an issue][new-issue]. When doing so, | ||||
| do use the issue templates - they provide a useful hint on what information to provide. | ||||
|  | ||||
| ## Translations | ||||
| Translations are managed through [Weblate], an online interface for managing language strings. This is synced | ||||
| automatically with GitHub, so please don't submit PRs adding/changing translations! | ||||
|  | ||||
| ## Developing | ||||
| In order to develop CC: Tweaked, you'll need to download the source code and then run it. This is a pretty simple | ||||
| process. When building on Windows, Use `gradlew.bat` instead of `./gradlew`. | ||||
|  | ||||
|  - **Clone the repository:** `git clone https://github.com/cc-tweaked/CC-Tweaked.git && cd CC-Tweaked` | ||||
|  - **Setup Forge:** `./gradlew build` | ||||
|  - **Run Minecraft:** `./gradlew runClient` (or run the `GradleStart` class from your IDE). | ||||
|  - **Optionally:** For small PRs (especially those only touching Lua code), it may be easier to use GitPod, which | ||||
|    provides a pre-configured environment: [](https://gitpod.io/#https://github.com/cc-tweaked/CC-Tweaked/) | ||||
|  | ||||
|    Do note you will need to download the mod after compiling to test. | ||||
|  | ||||
| If you want to run CC:T in a normal Minecraft instance, run `./gradlew build` and copy the `.jar` from `build/libs`. | ||||
| These commands may take a few minutes to run the first time, as the environment is set up, but should be much faster | ||||
| afterwards. | ||||
|  | ||||
| The following sections describe the more niche sections of CC: Tweaked's build system. Some bits of these are | ||||
| quite-complex, and (dare I say) over-engineered, so you may wish to ignore them. Well tested/documented PRs are always | ||||
| preferred (and I'd definitely recommend setting up the tooling if you're doing serious development work), but for | ||||
| small changes it can be a lot. | ||||
|  | ||||
| ### Code linters | ||||
| CC: Tweaked uses a couple of "linters" on its source code, to enforce a consistent style across the project. While these | ||||
| are run whenever you submit a PR, it's often useful to run this before committing. | ||||
|  | ||||
|  - **[Checkstyle]:** Checks Java code to ensure it is consistently formatted. This can be run with `./gradlew build` or | ||||
|    `./gradle check`. | ||||
|  - **[illuaminate]:** Checks Lua code for semantic and styleistic issues. See [the usage section][illuaminate-usage] for | ||||
|    how to download and run it. You may need to generate the Java documentation stubs (see "Documentation" below) for all | ||||
|    lints to pass. | ||||
|  | ||||
| ### Documentation | ||||
| When writing documentation for [CC: Tweaked's documentation website][docs], it may be useful to build the documentation | ||||
| and preview it yourself before submitting a PR. | ||||
|  | ||||
| Building all documentation is, sadly, a multi-stage process (though this is largely hidden by Gradle). First we need to | ||||
| convert Java doc-comments into Lua ones, we also generate some Javascript to embed. All of this is then finally fed into | ||||
| illuaminate, which spits out our HTML. | ||||
|  | ||||
| #### Setting up the tooling | ||||
| For various reasons, getting the environment set up to build documentation can be pretty complex. I'd quite like to | ||||
| automate this via Docker and/or nix in the future, but this needs to be done manually for now. | ||||
|  | ||||
| This tooling is only needed if you need to build the whole website. If you just want to generate the Lua stubs, you can | ||||
| skp this section. | ||||
|  - Install Node/npm and install our Node packages with `npm ci`. | ||||
|  - Install [illuaminate][illuaminate-usage] as described above. | ||||
|  | ||||
| #### Building documentation | ||||
| Gradle should be your entrypoint to building most documentation. There's two tasks which are of interest: | ||||
|  | ||||
|  - `./gradlew luaJavadoc` - Generate documentation stubs for Java methods. | ||||
|  - `./gradlew docWebsite` - Generate the whole website (including Javascript pages). The resulting HTML is stored at | ||||
|    `./build/docs/site/`. | ||||
|  | ||||
| #### Writing documentation | ||||
| illuaminate's documentation system is not currently documented (somewhat ironic), but is _largely_ the same as | ||||
| [ldoc][ldoc]. Documentation comments are written in Markdown, | ||||
|  | ||||
| Our markdown engine does _not_ support GitHub flavoured markdown, and so does not support all the features one might | ||||
| expect (such as tables). It is very much recommended that you build and preview the docs locally first. | ||||
|  | ||||
| ### Testing | ||||
| Thankfully running tests is much simpler than running the documentation generator! `./gradlew check` will run the | ||||
| entire test suite (and some additional bits of verification). | ||||
|  | ||||
| Before we get into writing tests, it's worth mentioning the various test suites that CC: Tweaked has: | ||||
|  - "Core" Java (`./src/test/java`): These test core bits of the mod which don't require any Minecraft interaction. | ||||
|    This includes the `@LuaFunction` system, file system code, etc... | ||||
|  | ||||
|    These tests are run by `./gradlew test`. | ||||
|  | ||||
|  - CraftOS (`./src/test/resources/test-rom/`): These tests are written in Lua, and ensure the Lua environment, libraries | ||||
|    and programs work as expected. These are (generally) written to be able to be run on emulators too, to provide some | ||||
|    sort of compliance test. | ||||
|  | ||||
|    These tests are run by the '"Core" Java' test suite, and so are also run with `./gradlew test`. | ||||
|  | ||||
|  - In-game (`./src/testMod/java/dan200/computercraft/ingame/`): These tests are run on an actual Minecraft server and client, | ||||
|    using [the same system Mojang do][mc-test]. The aim of these is to test in-game behaviour of blocks and peripherals. | ||||
|  | ||||
|    These are run by `./gradlew testClient` and `./gradlew testServer`. You may want to run the client under `xvfb-run` | ||||
|    or similar when running in a headless environment. | ||||
|  | ||||
| ## CraftOS tests | ||||
| CraftOS's tests are written using a test system called "mcfly", heavily inspired by [busted] (and thus RSpec). Groups of | ||||
| tests go inside `describe` blocks, and a single test goes inside `it`. | ||||
|  | ||||
| Assertions are generally written using `expect` (inspired by Hamcrest and the like). For instance, `expect(foo):eq("bar")` | ||||
| asserts that your variable `foo` is equal to the expected value `"bar"`. | ||||
|  | ||||
| [new-issue]: https://github.com/cc-tweaked/CC-Tweaked/issues/new/choose "Create a new issue" | ||||
| [community]: README.md#Community "Get in touch with the community." | ||||
| [checkstyle]: https://checkstyle.org/ | ||||
| [illuaminate]: https://github.com/SquidDev/illuaminate/ "Illuaminate on GitHub" | ||||
| [illuaminate-usage]: https://github.com/SquidDev/illuaminate/blob/master/README.md#usage "Installing Illuaminate" | ||||
| [weblate]: https://i18n.tweaked.cc/projects/cc-tweaked/minecraft/ "CC: Tweaked weblate instance" | ||||
| [docs]: https://tweaked.cc/ "CC: Tweaked documentation" | ||||
| [ldoc]: http://stevedonovan.github.io/ldoc/ "ldoc, a Lua documentation generator." | ||||
| [mc-test]: https://www.youtube.com/watch?v=vXaWOJTCYNg | ||||
| [busted]: https://github.com/Olivine-Labs/busted "busted: Elegant Lua unit testing." | ||||
							
								
								
									
										14
									
								
								LICENSE
									
									
									
									
									
								
							
							
						
						| @@ -19,14 +19,14 @@ Mod: The mod code designated by the present license, in source form, binary | ||||
| form, as obtained standalone, as part of a wider distribution or resulting from | ||||
| the compilation of the original or modified sources. | ||||
|  | ||||
| Dependency: Code required for the mod to work properly. This includes | ||||
| Dependency: Code required for the mod to work properly. This includes  | ||||
| dependencies required to compile the code as well as any file or modification | ||||
| that is explicitly or implicitly required for the mod to be working. | ||||
|  | ||||
| 1. Scope | ||||
| -------- | ||||
|  | ||||
| The present license is granted to any user of the mod. As a prerequisite, | ||||
| The present license is granted to any user of the mod. As a prerequisite,  | ||||
| a user must own a legally acquired copy of Minecraft | ||||
|  | ||||
| 2. Liability | ||||
| @@ -41,13 +41,13 @@ or misuse of this mod fall on the user. | ||||
| 3. Play rights | ||||
| -------------- | ||||
|  | ||||
| The user is allowed to install this mod on a Minecraft client or server and to play | ||||
| The user is allowed to install this mod on a Minecraft client or server and to play  | ||||
| without restriction. | ||||
|  | ||||
| 4. Modification rights | ||||
| ---------------------- | ||||
|  | ||||
| The user has the right to decompile the source code, look at either the | ||||
| The user has the right to decompile the source code, look at either the  | ||||
| decompiled version or the original source code, and to modify it. | ||||
|  | ||||
| 5. Distribution of original or modified copy rights | ||||
| @@ -61,10 +61,10 @@ include: | ||||
|    - patch to its source or binary files | ||||
|    - any copy of a portion of its binary source files | ||||
|  | ||||
| The user is allowed to redistribute this mod partially, in totality, or | ||||
| The user is allowed to redistribute this mod partially, in totality, or  | ||||
| included in a distribution. | ||||
|  | ||||
| When distributing binary files, the user must provide means to obtain its | ||||
| When distributing binary files, the user must provide means to obtain its  | ||||
| entire set of sources or modified sources at no cost. | ||||
|  | ||||
| All distributions of this mod must remain licensed under the CCPL. | ||||
| @@ -92,7 +92,7 @@ must be made available at no cost and remain licensed under the CCPL. | ||||
| --------------- | ||||
|  | ||||
| If you choose to contribute code or assets to be included in this mod, you | ||||
| agree that, if added to to the main repository at | ||||
| agree that, if added to to the main repository at  | ||||
| https://github.com/dan200/ComputerCraft, your contributions will be covered by | ||||
| this license, and that Daniel Ratcliffe will retain the right to re-license the | ||||
| mod, including your contributions, in part or in whole, under other licenses. | ||||
|   | ||||
							
								
								
									
										57
									
								
								README.md
									
									
									
									
									
								
							
							
						
						| @@ -1,54 +1,9 @@ | ||||
| #  | ||||
| [](https://github.com/cc-tweaked/CC-Tweaked/actions "Current build status") [][CurseForge] | ||||
| # CC:Restitched Patchwork | ||||
| # This is a Work In Progress Port | ||||
| *it runs and works-ish* | ||||
|  | ||||
| CC: Tweaked is a mod for Minecraft which adds programmable computers, turtles and more to the game. A fork of the | ||||
| much-beloved [ComputerCraft], it continues its legacy with better performance, stability, and a wealth of new features. | ||||
| ## Reached Parity with CC:T 1.92.0 | ||||
|  | ||||
| CC: Tweaked can be installed from [CurseForge] or [Modrinth]. It requires the [Minecraft Forge][forge] mod loader, but | ||||
| [versions are available for Fabric][ccrestitched]. | ||||
| THis is just a quick patchwork of my attempts at getting CC:R up to date with CC:T | ||||
|  | ||||
| ## Contributing | ||||
| Any contribution is welcome, be that using the mod, reporting bugs or contributing code. If you want to get started | ||||
| developing the mod, [check out the instructions here](CONTRIBUTING.md#developing). | ||||
|  | ||||
| ## Community | ||||
| If you need help getting started with CC: Tweaked, want to show off your latest project, or just want to chat about | ||||
| ComputerCraft we have a [forum](https://forums.computercraft.cc/) and [Discord guild](https://discord.computercraft.cc)! | ||||
| There's also a fairly populated, albeit quiet [IRC channel](http://webchat.esper.net/?channels=computercraft), if that's | ||||
| more your cup of tea. | ||||
|  | ||||
| We also host fairly comprehensive documentation at [tweaked.cc](https://tweaked.cc/ "The CC: Tweaked website"). | ||||
|  | ||||
| ## Using | ||||
| CC: Tweaked is hosted on my maven repo, and so is relatively simple to depend on. You may wish to add a soft (or hard) | ||||
| dependency in your `mods.toml` file, with the appropriate version bounds, to ensure that API functionality you depend | ||||
| on is present. | ||||
|  | ||||
| ```groovy | ||||
| repositories { | ||||
|   maven { | ||||
|     url 'https://squiddev.cc/maven/' | ||||
|     content { | ||||
|       includeGroup 'org.squiddev' | ||||
|     } | ||||
|   } | ||||
| } | ||||
|  | ||||
| dependencies { | ||||
|   compileOnly fg.deobf("org.squiddev:cc-tweaked-${mc_version}:${cct_version}:api") | ||||
|   runtimeOnly fg.deobf("org.squiddev:cc-tweaked-${mc_version}:${cct_version}") | ||||
| } | ||||
| ``` | ||||
|  | ||||
| You should also be careful to only use classes within the `dan200.computercraft.api` package. Non-API classes are | ||||
| subject to change at any point. If you depend on functionality outside the API, file an issue, and we can look into | ||||
| exposing more features. | ||||
|  | ||||
| We bundle the API sources with the jar, so documentation should be easily viewable within your editor. Alternatively, | ||||
| the generated documentation [can be browsed online](https://tweaked.cc/javadoc/). | ||||
|  | ||||
| [computercraft]: https://github.com/dan200/ComputerCraft "ComputerCraft on GitHub" | ||||
| [curseforge]: https://minecraft.curseforge.com/projects/cc-tweaked "Download CC: Tweaked from CurseForge" | ||||
| [modrinth]: https://modrinth.com/mod/gu7yAYhd "Download CC: Tweaked from Modrinth" | ||||
| [forge]: https://files.minecraftforge.net/ "Download Minecraft Forge." | ||||
| [ccrestitched]: https://www.curseforge.com/minecraft/mc-mods/cc-restitched "Download CC: Restitched from CurseForge" | ||||
| The changelog is located at [PatchWork.md](patchwork.md) | ||||
							
								
								
									
										602
									
								
								build.gradle
									
									
									
									
									
								
							
							
						
						| @@ -1,113 +1,15 @@ | ||||
| plugins { | ||||
|     id "checkstyle" | ||||
|     id "jacoco" | ||||
|     id "maven-publish" | ||||
|     id "org.cadixdev.licenser" version "0.6.1" | ||||
|     id "com.matthewprenger.cursegradle" version "1.4.0" | ||||
|     id "com.github.breadmoirai.github-release" version "2.2.12" | ||||
|     id "org.jetbrains.kotlin.jvm" version "1.7.0" | ||||
|     id "com.modrinth.minotaur" version "2.+" | ||||
|     id "net.minecraftforge.gradle" version "5.1.+" | ||||
|     id "org.spongepowered.mixin" version "0.7.+" | ||||
|     id "org.parchmentmc.librarian.forgegradle" version "1.+" | ||||
|     id "com.github.johnrengelman.shadow" version "7.1.2" | ||||
|     id 'fabric-loom' version '0.5-SNAPSHOT' | ||||
|     id 'maven-publish' | ||||
| } | ||||
|  | ||||
| import org.apache.tools.ant.taskdefs.condition.Os | ||||
| sourceCompatibility = JavaVersion.VERSION_1_8 | ||||
| targetCompatibility = JavaVersion.VERSION_1_8 | ||||
|  | ||||
| version = mod_version | ||||
|  | ||||
| group = "org.squiddev" | ||||
| archivesBaseName = "cc-tweaked-${mc_version}" | ||||
|  | ||||
| def javaVersion = JavaLanguageVersion.of(8) | ||||
| java { | ||||
|     toolchain { | ||||
|         languageVersion = javaVersion | ||||
|     } | ||||
|  | ||||
|     withSourcesJar() | ||||
|     withJavadocJar() | ||||
|     registerFeature("extraMods") { usingSourceSet(sourceSets.main) } | ||||
| } | ||||
|  | ||||
| sourceSets { | ||||
|     main.resources { | ||||
|         srcDir 'src/generated/resources' | ||||
|     } | ||||
|  | ||||
|     testMod {} | ||||
| } | ||||
|  | ||||
| minecraft { | ||||
|     runs { | ||||
|         all { | ||||
|             property 'forge.logging.markers', 'REGISTRIES' | ||||
|             property 'forge.logging.console.level', 'debug' | ||||
|             forceExit = false | ||||
|  | ||||
|             mods { | ||||
|                 computercraft { | ||||
|                     source sourceSets.main | ||||
|                 } | ||||
|             } | ||||
|  | ||||
|             arg "-mixin.config=computercraft.mixins.json" | ||||
|         } | ||||
|  | ||||
|         client { | ||||
|             workingDirectory project.file('run') | ||||
|         } | ||||
|  | ||||
|         server { | ||||
|             workingDirectory project.file("run/server") | ||||
|             arg "--nogui" | ||||
|         } | ||||
|  | ||||
|         data { | ||||
|             workingDirectory project.file('run') | ||||
|             args '--mod', 'computercraft', '--all', '--output', file('src/generated/resources/'), '--existing', file('src/main/resources/') | ||||
|         } | ||||
|  | ||||
|         testClient { | ||||
|             workingDirectory project.file('test-files/client') | ||||
|             parent runs.client | ||||
|  | ||||
|             mods { | ||||
|                 cctest { | ||||
|                     source sourceSets.testMod | ||||
|                 } | ||||
|             } | ||||
|         } | ||||
|  | ||||
|         testServer { | ||||
|             workingDirectory project.file('test-files/server') | ||||
|             parent runs.server | ||||
|  | ||||
|             property("cctest.run", "true") | ||||
|             property("forge.logging.console.level", "info") | ||||
|  | ||||
|             mods { | ||||
|                 cctest { | ||||
|                     source sourceSets.testMod | ||||
|                 } | ||||
|             } | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     mappings channel: 'parchment', version: "${mapping_version}-${mc_version}" | ||||
|  | ||||
|     accessTransformer file('src/main/resources/META-INF/accesstransformer.cfg') | ||||
|     accessTransformer file('src/testMod/resources/META-INF/accesstransformer.cfg') | ||||
| } | ||||
|  | ||||
| mixin { | ||||
|     add sourceSets.main, 'computercraft.mixins.refmap.json' | ||||
| } | ||||
|  | ||||
| reobf { | ||||
|     shadowJar {} | ||||
| } | ||||
| group = "dan200.computercraft" | ||||
| archivesBaseName = "cc-tweaked-fabric-${mc_version}" | ||||
|  | ||||
| repositories { | ||||
|     mavenCentral() | ||||
| @@ -118,477 +20,101 @@ repositories { | ||||
| } | ||||
|  | ||||
| configurations { | ||||
|     shade { transitive = false } | ||||
|     implementation.extendsFrom shade | ||||
|     cctJavadoc | ||||
|  | ||||
|     testModImplementation.extendsFrom(implementation) | ||||
|     testModImplementation.extendsFrom(testImplementation) | ||||
|     compile.extendsFrom shade | ||||
| } | ||||
|  | ||||
| dependencies { | ||||
|     checkstyle "com.puppycrawl.tools:checkstyle:8.25" | ||||
|     minecraft "com.mojang:minecraft:${mc_version}" | ||||
|     mappings "net.fabricmc:yarn:${mc_version}+build.${mappings_version}:v2" | ||||
|     modImplementation "net.fabricmc:fabric-loader:${fabric_loader_version}" | ||||
|     modImplementation "net.fabricmc.fabric-api:fabric-api:${fabric_api_version}" | ||||
|  | ||||
|     minecraft "net.minecraftforge:forge:${mc_version}-${forge_version}" | ||||
|     annotationProcessor 'org.spongepowered:mixin:0.8.4:processor' | ||||
|     modImplementation "me.shedaniel.cloth:config-2:${cloth_config_version}" | ||||
|     modImplementation "io.github.prospector:modmenu:${modmenu_version}" | ||||
|  | ||||
|     extraModsCompileOnly fg.deobf("mezz.jei:jei-1.16.5:7.7.0.104:api") | ||||
|     extraModsRuntimeOnly fg.deobf("mezz.jei:jei-1.16.5:7.7.0.104") | ||||
|     modApi "me.shedaniel.cloth.api:cloth-utils-v1:${project.cloth_api_version}" | ||||
|     include "me.shedaniel.cloth.api:cloth-utils-v1:${project.cloth_api_version}" | ||||
|  | ||||
|     extraModsCompileOnly fg.deobf("com.blamejared.crafttweaker:CraftTweaker-1.16.5:7.1.0.313") | ||||
|     extraModsCompileOnly fg.deobf("commoble.morered:morered-1.16.5:2.1.1.0") | ||||
|     implementation "blue.endless:jankson:${jankson_version}" | ||||
|     implementation 'com.google.code.findbugs:jsr305:3.0.2' | ||||
|  | ||||
|     shade 'org.squiddev:Cobalt:0.5.5' | ||||
|     compileOnly 'com.google.auto.service:auto-service:1.0-rc7' | ||||
|     annotationProcessor 'com.google.auto.service:auto-service:1.0-rc7' | ||||
|  | ||||
|     testImplementation 'org.junit.jupiter:junit-jupiter-api:5.7.0' | ||||
|     testImplementation 'org.junit.jupiter:junit-jupiter-params:5.7.0' | ||||
|     testRuntimeOnly 'org.junit.jupiter:junit-jupiter-engine:5.7.0' | ||||
|     testImplementation 'org.hamcrest:hamcrest:2.2' | ||||
|     testImplementation 'org.jetbrains.kotlin:kotlin-stdlib-jdk8:1.7.0' | ||||
|     testImplementation 'org.jetbrains.kotlinx:kotlinx-coroutines-core:1.6.2' | ||||
|     include "me.shedaniel.cloth:config-2:${cloth_config_version}" | ||||
|     include "blue.endless:jankson:${jankson_version}" | ||||
|     include 'javax.vecmath:vecmath:1.5.2' | ||||
|  | ||||
|     testModImplementation sourceSets.main.output | ||||
|     compile 'javax.vecmath:vecmath:1.5.2' | ||||
|  | ||||
|     cctJavadoc 'cc.tweaked:cct-javadoc:1.4.6' | ||||
|     shade 'org.squiddev:Cobalt:0.5.1-SNAPSHOT' | ||||
|  | ||||
|     testImplementation 'org.junit.jupiter:junit-jupiter-api:5.1.0' | ||||
|     testRuntimeOnly 'org.junit.jupiter:junit-jupiter-engine:5.1.0' | ||||
|  | ||||
|     modRuntime "me.shedaniel:RoughlyEnoughItems-api:5.2.10" | ||||
|     modRuntime "me.shedaniel:RoughlyEnoughItems:5.2.10" | ||||
| } | ||||
|  | ||||
| // Compile tasks | ||||
|  | ||||
| javadoc { | ||||
|     include "dan200/computercraft/api/**/*.java" | ||||
| } | ||||
|  | ||||
| def apiJar = tasks.register("apiJar", Jar.class) { | ||||
|     archiveClassifier.set("api") | ||||
|     from(sourceSets.main.output) { | ||||
|         include "dan200/computercraft/api/**/*" | ||||
|     } | ||||
| } | ||||
| assemble.dependsOn(apiJar) | ||||
|  | ||||
| def luaJavadoc = tasks.register("luaJavadoc", Javadoc.class) { | ||||
|     description "Generates documentation for Java-side Lua functions." | ||||
|     group "documentation" | ||||
|  | ||||
|     source = sourceSets.main.allJava | ||||
|     destinationDir = file("${project.docsDir}/luaJavadoc") | ||||
|     classpath = sourceSets.main.compileClasspath | ||||
|  | ||||
|     options.docletpath = configurations.cctJavadoc.files as List | ||||
|     options.doclet = "cc.tweaked.javadoc.LuaDoclet" | ||||
|     options.noTimestamp = false | ||||
|  | ||||
|     javadocTool = javaToolchains.javadocToolFor { | ||||
|         languageVersion = JavaLanguageVersion.of(11) | ||||
|     } | ||||
| } | ||||
|  | ||||
| jar { | ||||
|     finalizedBy("reobfJar") | ||||
|  | ||||
|     archiveClassifier.set("slim") | ||||
|     manifest { | ||||
|         attributes([ | ||||
|             "Specification-Title"     : "computercraft", | ||||
|             "Specification-Vendor"    : "SquidDev", | ||||
|             "Specification-Version"   : "1", | ||||
|             "Implementation-Title"    : "CC: Tweaked", | ||||
|             "Implementation-Version"  : "${mod_version}", | ||||
|             "Implementation-Vendor"   : "SquidDev", | ||||
|             "Implementation-Timestamp": new Date().format("yyyy-MM-dd'T'HH:mm:ssZ"), | ||||
|             "MixinConfigs"            : "computercraft.mixins.json", | ||||
|         ]) | ||||
|     } | ||||
| } | ||||
|  | ||||
| shadowJar { | ||||
|     finalizedBy("reobfShadowJar") | ||||
|  | ||||
|     archiveClassifier.set("") | ||||
|     configurations = [project.configurations.shade] | ||||
|     relocate("org.squiddev.cobalt", "cc.tweaked.internal.cobalt") | ||||
|     minimize() | ||||
| } | ||||
|  | ||||
| assemble.dependsOn("shadowJar") | ||||
|  | ||||
| [ | ||||
|     tasks.named("compileJava", JavaCompile.class), | ||||
|     tasks.named("compileTestJava", JavaCompile.class), | ||||
|     tasks.named("compileTestModJava", JavaCompile.class) | ||||
| ].forEach { | ||||
|     it.configure { | ||||
|         options.compilerArgs << "-Xlint" << "-Xlint:-processing" | ||||
| sourceSets { | ||||
|     main { | ||||
|         java { | ||||
|             exclude 'dan200/computercraft/shared/integration' | ||||
|         } | ||||
|     } | ||||
| } | ||||
|  | ||||
| processResources { | ||||
|     inputs.property "version", mod_version | ||||
|     inputs.property "mcversion", mc_version | ||||
|  | ||||
|     def hash = 'none' | ||||
|     Set<String> contributors = [] | ||||
|     try { | ||||
|         hash = ["git", "-C", projectDir, "rev-parse", "HEAD"].execute().text.trim() | ||||
|  | ||||
|         def blacklist = ['GitHub', 'Daniel Ratcliffe', 'Weblate'] | ||||
|  | ||||
|         // Extract all authors, commiters and co-authors from the git log. | ||||
|         def authors = ["git", "-C", projectDir, "log", "--format=tformat:%an <%ae>%n%cn <%ce>%n%(trailers:key=Co-authored-by,valueonly)"] | ||||
|             .execute().text.readLines().unique() | ||||
|  | ||||
|         // We now pass this through git's mailmap to de-duplicate some authors. | ||||
|         def remapAuthors = ["git", "check-mailmap", "--stdin"].execute() | ||||
|         remapAuthors.withWriter { stdin -> | ||||
|             if (stdin !instanceof BufferedWriter) stdin = new BufferedWriter(stdin) | ||||
|  | ||||
|             authors.forEach { | ||||
|                 if (it == "") return | ||||
|                 if (!it.endsWith(">")) it += ">" // Some commits have broken Co-Authored-By lines! | ||||
|                 stdin.writeLine(it) | ||||
|             } | ||||
|             stdin.close() | ||||
|         } | ||||
|  | ||||
|         // And finally extract out the actual name. | ||||
|         def emailRegex = ~/^([^<]+) <.+>$/ | ||||
|         remapAuthors.text.readLines().forEach { | ||||
|             def matcher = it =~ emailRegex | ||||
|             matcher.find() | ||||
|             def name = matcher.group(1) | ||||
|             if (!blacklist.contains(name)) contributors.add(name) | ||||
|         } | ||||
|     } catch (Exception e) { | ||||
|         e.printStackTrace() | ||||
|     } | ||||
|     inputs.property "commithash", hash | ||||
|     duplicatesStrategy = DuplicatesStrategy.INCLUDE | ||||
|     inputs.property "version", project.version | ||||
|  | ||||
|     from(sourceSets.main.resources.srcDirs) { | ||||
|         include 'META-INF/mods.toml' | ||||
|         include 'data/computercraft/lua/rom/help/credits.txt' | ||||
|  | ||||
|         expand 'version': mod_version, | ||||
|             'mcversion': mc_version, | ||||
|             'gitcontributors': contributors.sort(false, String.CASE_INSENSITIVE_ORDER).join('\n') | ||||
|         include "fabric.mod.json" | ||||
|         expand "version": project.version | ||||
|     } | ||||
|  | ||||
|     from(sourceSets.main.resources.srcDirs) { | ||||
|         exclude 'META-INF/mods.toml' | ||||
|         exclude 'data/computercraft/lua/rom/help/credits.txt' | ||||
|         exclude "fabric.mod.json" | ||||
|     } | ||||
| } | ||||
|  | ||||
| sourcesJar { | ||||
|     duplicatesStrategy = DuplicatesStrategy.INCLUDE | ||||
| // ensure that the encoding is set to UTF-8, no matter what the system default is | ||||
| // this fixes some edge cases with special characters not displaying correctly | ||||
| // see http://yodaconditions.net/blog/fix-for-java-file-encoding-problems-with-gradle.html | ||||
| tasks.withType(JavaCompile) { | ||||
|     options.encoding = "UTF-8" | ||||
| } | ||||
|  | ||||
| // Web tasks | ||||
|  | ||||
| List<String> mkCommand(String command) { | ||||
|     return Os.isFamily(Os.FAMILY_WINDOWS) ? ["cmd", "/c", command] : ["sh", "-c", command] | ||||
| // Loom will automatically attach sourcesJar to a RemapSourcesJar task and to the "build" task | ||||
| // if it is present. | ||||
| // If you remove this task, sources will not be generated. | ||||
| task sourcesJar(type: Jar, dependsOn: classes) { | ||||
|     classifier = "sources" | ||||
|     from sourceSets.main.allSource | ||||
| } | ||||
|  | ||||
| def rollup = tasks.register("rollup", Exec.class) { | ||||
|     group = "build" | ||||
|     description = "Bundles JS into rollup" | ||||
| jar { | ||||
|     from "LICENSE" | ||||
|  | ||||
|     inputs.files(fileTree("src/web")).withPropertyName("sources") | ||||
|     inputs.file("package-lock.json").withPropertyName("package-lock.json") | ||||
|     inputs.file("tsconfig.json").withPropertyName("Typescript config") | ||||
|     inputs.file("rollup.config.js").withPropertyName("Rollup config") | ||||
|     outputs.file("$buildDir/rollup/index.js").withPropertyName("output") | ||||
|  | ||||
|     commandLine mkCommand('"node_modules/.bin/rollup" --config rollup.config.js') | ||||
| } | ||||
|  | ||||
| def illuaminateDocs = tasks.register("illuaminateDocs", Exec.class) { | ||||
|     group = "documentation" | ||||
|     description = "Generates docs using Illuaminate" | ||||
|     dependsOn(rollup, luaJavadoc) | ||||
|  | ||||
|     inputs.files(fileTree("doc")).withPropertyName("docs") | ||||
|     inputs.files(fileTree("src/main/resources/data/computercraft/lua/rom")).withPropertyName("lua rom") | ||||
|     inputs.file("illuaminate.sexp").withPropertyName("illuaminate.sexp") | ||||
|     inputs.dir("$buildDir/docs/luaJavadoc") | ||||
|     inputs.file("$buildDir/rollup/index.js").withPropertyName("scripts") | ||||
|     inputs.file("src/web/styles.css").withPropertyName("styles") | ||||
|     outputs.dir("$buildDir/docs/lua") | ||||
|  | ||||
|     commandLine mkCommand('"bin/illuaminate" doc-gen') | ||||
| } | ||||
|  | ||||
| def jsxDocs = tasks.register("jsxDocs", Exec) { | ||||
|     group = "documentation" | ||||
|     description = "Post-processes documentation to statically render some dynamic content." | ||||
|     dependsOn(illuaminateDocs) | ||||
|  | ||||
|     inputs.files(fileTree("src/web")).withPropertyName("sources") | ||||
|     inputs.file("src/generated/export/index.json").withPropertyName("export") | ||||
|     inputs.file("package-lock.json").withPropertyName("package-lock.json") | ||||
|     inputs.file("tsconfig.json").withPropertyName("Typescript config") | ||||
|     inputs.files(fileTree("$buildDir/docs/lua")) | ||||
|     outputs.dir("$buildDir/docs/site") | ||||
|  | ||||
|     commandLine mkCommand('"node_modules/.bin/ts-node" --esm src/web/transform.tsx') | ||||
| } | ||||
|  | ||||
| def docWebsite = tasks.register("docWebsite", Copy.class) { | ||||
|     group = "documentation" | ||||
|     description = "Copy additional assets to the website directory." | ||||
|     dependsOn(jsxDocs) | ||||
|  | ||||
|     from('doc') { | ||||
|         include 'logo.png' | ||||
|         include 'images/**' | ||||
|     } | ||||
|     from("$buildDir/rollup") { | ||||
|         exclude 'index.js' | ||||
|     } | ||||
|     from("$buildDir/docs/lua") { | ||||
|         exclude '**/*.html' | ||||
|     } | ||||
|     from("src/generated/export/items") { | ||||
|         into("images/items") | ||||
|     } | ||||
|  | ||||
|     into "${project.docsDir}/site" | ||||
| } | ||||
|  | ||||
| // Check tasks | ||||
|  | ||||
| test { | ||||
|     useJUnitPlatform() | ||||
|     testLogging { | ||||
|         events "skipped", "failed" | ||||
|     } | ||||
| } | ||||
|  | ||||
| jacocoTestReport { | ||||
|     dependsOn('test') | ||||
|     reports { | ||||
|         xml.required = true | ||||
|         html.required = true | ||||
|     } | ||||
| } | ||||
|  | ||||
| test.finalizedBy("jacocoTestReport") | ||||
|  | ||||
| license { | ||||
|     header = file('config/license/main.txt') | ||||
|     lineEnding = '\n' | ||||
|     newLine = false | ||||
|  | ||||
|     properties { | ||||
|         year = Calendar.getInstance().get(Calendar.YEAR) | ||||
|     } | ||||
|  | ||||
|     include("**/*.java") // We could apply to Kotlin, but for now let's not | ||||
|     matching("dan200/computercraft/api/**") { | ||||
|         header = file('config/license/api.txt') | ||||
|     } | ||||
| } | ||||
|  | ||||
| check.dependsOn("licenseCheck") | ||||
|  | ||||
| def setupServer = tasks.register("setupServer", Copy.class) { | ||||
|     group "test server" | ||||
|     description "Sets up the environment for the test server." | ||||
|  | ||||
|     from("src/testMod/server-files") { | ||||
|         include "eula.txt" | ||||
|         include "server.properties" | ||||
|     } | ||||
|     into "test-files/server" | ||||
| } | ||||
|  | ||||
| def testServerClassDumpDir = new File(buildDir, "jacocoClassDump/runTestServer") | ||||
|  | ||||
| def testServer = tasks.register("testServer", JavaExec.class) { | ||||
|     group("In-game tests") | ||||
|     description("Runs tests on a temporary Minecraft instance.") | ||||
|     dependsOn(setupServer, "cleanTestServer") | ||||
|     finalizedBy("jacocoTestServerReport") | ||||
|  | ||||
|     // Copy from runTestServer. We do it in this slightly odd way as runTestServer | ||||
|     // isn't created until the task is configured (which is no good for us). | ||||
|     JavaExec exec = tasks.getByName("runTestServer") | ||||
|     dependsOn(exec.getDependsOn()) | ||||
|     exec.copyTo(it) | ||||
|     setClasspath(exec.getClasspath()) | ||||
|     mainClass = exec.mainClass | ||||
|     javaLauncher = exec.javaLauncher | ||||
|     setArgs(exec.getArgs()) | ||||
|  | ||||
|     // Jacoco and modlauncher don't play well together as the classes loaded in-game don't | ||||
|     // match up with those written to disk. We get Jacoco to dump all classes to disk, and | ||||
|     // use that when generating the report. | ||||
|     jacoco.applyTo(it) | ||||
|     it.jacoco.setIncludes(["dan200.computercraft.*"]) | ||||
|     it.jacoco.setClassDumpDir(testServerClassDumpDir) | ||||
|     outputs.dir(testServerClassDumpDir) | ||||
|     // Older versions of modlauncher don't include a protection domain (and thus no code | ||||
|     // source). Jacoco skips such classes by default, so we need to explicitly include them. | ||||
|     it.jacoco.setIncludeNoLocationClasses(true) | ||||
| } | ||||
|  | ||||
| tasks.register("jacocoTestServerReport", JacocoReport.class) { | ||||
|     group("In-game tests") | ||||
|     description("Generate coverage reports for testServer") | ||||
|     dependsOn(testServer) | ||||
|  | ||||
|     executionData(new File(buildDir, "jacoco/testServer.exec")) | ||||
|     sourceDirectories.from(sourceSets.main.allJava.srcDirs) | ||||
|     classDirectories.from(testServerClassDumpDir) | ||||
|  | ||||
|     reports { | ||||
|         xml.enabled true | ||||
|         html.enabled true | ||||
|     } | ||||
| } | ||||
|  | ||||
| check.dependsOn(testServer) | ||||
|  | ||||
| // Upload tasks | ||||
|  | ||||
| def checkRelease = tasks.register("checkRelease") { | ||||
|     group "upload" | ||||
|     description "Verifies that everything is ready for a release" | ||||
|  | ||||
|     inputs.property "version", mod_version | ||||
|     inputs.file("src/main/resources/data/computercraft/lua/rom/help/changelog.md") | ||||
|     inputs.file("src/main/resources/data/computercraft/lua/rom/help/whatsnew.md") | ||||
|  | ||||
|     doLast { | ||||
|         def ok = true | ||||
|  | ||||
|         // Check we're targetting the current version | ||||
|         def whatsnew = new File(projectDir, "src/main/resources/data/computercraft/lua/rom/help/whatsnew.md").readLines() | ||||
|         if (whatsnew[0] != "New features in CC: Tweaked $mod_version") { | ||||
|             ok = false | ||||
|             project.logger.error("Expected `whatsnew.md' to target $mod_version.") | ||||
|         } | ||||
|  | ||||
|         // Check "read more" exists and trim it | ||||
|         def idx = whatsnew.findIndexOf { it == 'Type "help changelog" to see the full version history.' } | ||||
|         if (idx == -1) { | ||||
|             ok = false | ||||
|             project.logger.error("Must mention the changelog in whatsnew.md") | ||||
|         } else { | ||||
|             whatsnew = whatsnew.getAt(0..<idx) | ||||
|         } | ||||
|  | ||||
|         // Check whatsnew and changelog match. | ||||
|         def versionChangelog = "# " + whatsnew.join("\n") | ||||
|         def changelog = new File(projectDir, "src/main/resources/data/computercraft/lua/rom/help/changelog.md").getText() | ||||
|         if (!changelog.startsWith(versionChangelog)) { | ||||
|             ok = false | ||||
|             project.logger.error("whatsnew and changelog are not in sync") | ||||
|         } | ||||
|  | ||||
|         if (!ok) throw new IllegalStateException("Could not check release") | ||||
|     } | ||||
| } | ||||
| check.dependsOn(checkRelease) | ||||
|  | ||||
| def isStable = true | ||||
|  | ||||
| curseforge { | ||||
|     apiKey = project.hasProperty('curseForgeApiKey') ? project.curseForgeApiKey : '' | ||||
|     project { | ||||
|         id = '282001' | ||||
|         releaseType = isStable ? 'release' : 'alpha' | ||||
|         changelog = "Release notes can be found on the GitHub repository (https://github.com/cc-tweaked/CC-Tweaked/releases/tag/v${mc_version}-${mod_version})." | ||||
|         mainArtifact(shadowJar) | ||||
|  | ||||
|         addGameVersion "${mc_version}" | ||||
|     } | ||||
| } | ||||
|  | ||||
| modrinth { | ||||
|     token = project.hasProperty('modrinthApiKey') ? project.getProperty('modrinthApiKey') : '' | ||||
|     projectId = 'gu7yAYhd' | ||||
|     versionNumber = "${project.mc_version}-${project.mod_version}" | ||||
|     versionType = isStable ? 'release' : 'alpha' | ||||
|     uploadFile = shadowJar | ||||
|     gameVersions = [project.mc_version] | ||||
|     changelog = "Release notes can be found on the [GitHub repository](https://github.com/cc-tweaked/CC-Tweaked/releases/tag/v${mc_version}-${mod_version})." | ||||
|     from configurations.shade.collect { it.isDirectory() ? it : zipTree(it) } | ||||
| } | ||||
|  | ||||
| // configure the maven publication | ||||
| publishing { | ||||
|     publications { | ||||
|         maven(MavenPublication) { | ||||
|             from components.java | ||||
|             artifact(apiJar) | ||||
|             fg.component(it) | ||||
|  | ||||
|             pom { | ||||
|                 name = 'CC: Tweaked' | ||||
|                 description = 'CC: Tweaked is a fork of ComputerCraft, adding programmable computers, turtles and more to Minecraft.' | ||||
|                 url = 'https://github.com/cc-tweaked/CC-Tweaked' | ||||
|  | ||||
|                 scm { | ||||
|                     url = 'https://github.com/cc-tweaked/CC-Tweaked.git' | ||||
|                 } | ||||
|  | ||||
|                 issueManagement { | ||||
|                     system = 'github' | ||||
|                     url = 'https://github.com/cc-tweaked/CC-Tweaked/issues' | ||||
|                 } | ||||
|  | ||||
|                 licenses { | ||||
|                     license { | ||||
|                         name = 'ComputerCraft Public License, Version 1.0' | ||||
|                         url = 'https://github.com/cc-tweaked/CC-Tweaked/blob/mc-1.16.x/LICENSE' | ||||
|                     } | ||||
|                 } | ||||
|         mavenJava(MavenPublication) { | ||||
|             // add all the jars that should be included when publishing to maven | ||||
|             artifact(remapJar) { | ||||
|                 builtBy remapJar | ||||
|             } | ||||
|             artifact(sourcesJar) { | ||||
|                 builtBy remapSourcesJar | ||||
|             } | ||||
|         } | ||||
|     } | ||||
|  | ||||
|     // select the repositories you want to publish to | ||||
|     repositories { | ||||
|         if (project.hasProperty("mavenUser")) { | ||||
|             maven { | ||||
|                 name = "SquidDev" | ||||
|                 url = "https://squiddev.cc/maven" | ||||
|                 credentials { | ||||
|                     username = project.property("mavenUser") as String | ||||
|                     password = project.property("mavenPass") as String | ||||
|                 } | ||||
|             } | ||||
|         } | ||||
|         // uncomment to publish to the local maven | ||||
|         // mavenLocal() | ||||
|     } | ||||
| } | ||||
|  | ||||
| githubRelease { | ||||
|     token project.hasProperty('githubApiKey') ? project.githubApiKey : '' | ||||
|     owner 'cc-tweaked' | ||||
|     repo 'CC-Tweaked' | ||||
|     targetCommitish.set(project.provider({ | ||||
|         try { | ||||
|             return ["git", "-C", projectDir, "rev-parse", "--abbrev-ref", "HEAD"].execute().text.trim() | ||||
|         } catch (Exception e) { | ||||
|             e.printStackTrace() | ||||
|         } | ||||
|         return "master" | ||||
|     })) | ||||
|  | ||||
|     tagName "v${mc_version}-${mod_version}" | ||||
|     releaseName "[${mc_version}] ${mod_version}" | ||||
|     body.set(project.provider({ | ||||
|         "## " + new File(projectDir, "src/main/resources/data/computercraft/lua/rom/help/whatsnew.md") | ||||
|             .readLines() | ||||
|             .takeWhile { it != 'Type "help changelog" to see the full version history.' } | ||||
|             .join("\n").trim() | ||||
|     })) | ||||
|     prerelease !isStable | ||||
| } | ||||
|  | ||||
| def uploadTasks = ["publish", "curseforge", "modrinth", "githubRelease"] | ||||
| uploadTasks.forEach { tasks.named(it) { dependsOn(checkRelease) } } | ||||
|  | ||||
| tasks.register("uploadAll") { | ||||
|     group = "upload" | ||||
|     description = "Uploads to all repositories (Maven, Curse, Modrinth, GitHub release)" | ||||
|     dependsOn(uploadTasks) | ||||
| } | ||||
|   | ||||
| @@ -2488,4 +2488,4 @@ | ||||
|       </option> | ||||
|     </inspection_tool> | ||||
|   </profile> | ||||
| </component> | ||||
| </component> | ||||
| @@ -58,4 +58,4 @@ | ||||
|       <option name="CONTINUATION_INDENT_SIZE" value="4" /> | ||||
|     </indentOptions> | ||||
|   </codeStyleSettings> | ||||
| </code_scheme> | ||||
| </code_scheme> | ||||
| @@ -1,170 +0,0 @@ | ||||
| <?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_loc}/suppressions.xml" /> | ||||
|     </module> | ||||
|  | ||||
|     <module name="BeforeExecutionExclusionFileFilter"> | ||||
|         <property name="fileNamePattern" value="render_old"/> | ||||
|     </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"> | ||||
|             <property name="checkHtml" value="false" /> | ||||
|         </module> | ||||
|         <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" /> | ||||
|         <module name="MemberName" /> | ||||
|         <module name="MethodName" /> | ||||
|         <module name="MethodTypeParameterName" /> | ||||
|         <module name="PackageName"> | ||||
|             <property name="format" value="^dan200\.computercraft(\.[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="ignoreEnhancedForColon" value="false" /> | ||||
|             <!-- Allow empty functions --> | ||||
|             <property name="allowEmptyLambdas" value="true" /> | ||||
|             <property name="allowEmptyMethods" value="true" /> | ||||
|             <property name="allowEmptyConstructors" value="true" /> | ||||
|  | ||||
|             <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> | ||||
| @@ -1,12 +0,0 @@ | ||||
| <?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" /> | ||||
|  | ||||
|     <!-- The commands API is documented in Lua. --> | ||||
|     <suppress checks="SummaryJavadocCheck" files=".*[\\/]CommandAPI.java" /> | ||||
| </suppressions> | ||||
| @@ -1,8 +0,0 @@ | ||||
| FROM gitpod/workspace-base | ||||
|  | ||||
| USER gitpod | ||||
|  | ||||
| RUN sudo apt-get -q update \ | ||||
|  && sudo apt-get install -yq openjdk-8-jdk openjdk-16-jdk python3-pip npm \ | ||||
|  && sudo pip3 install pre-commit \ | ||||
|  && sudo update-java-alternatives --set java-1.8.0-openjdk-amd64 | ||||
| @@ -1,3 +0,0 @@ | ||||
| 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. | ||||
| @@ -1,3 +0,0 @@ | ||||
| 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,56 +0,0 @@ | ||||
| # See https://pre-commit.com for more information | ||||
| # See https://pre-commit.com/hooks.html for more hooks | ||||
| repos: | ||||
| - repo: https://github.com/pre-commit/pre-commit-hooks | ||||
|   rev: v4.0.1 | ||||
|   hooks: | ||||
|   - id: trailing-whitespace | ||||
|   - id: end-of-file-fixer | ||||
|   - id: check-merge-conflict | ||||
|  | ||||
|   # Quick syntax checkers | ||||
|   - id: check-xml | ||||
|   - id: check-yaml | ||||
|   - id: check-toml | ||||
|   - id: check-json | ||||
|     exclude: "tsconfig\\.json$" | ||||
|  | ||||
| - repo: https://github.com/editorconfig-checker/editorconfig-checker.python | ||||
|   rev: 2.3.54 | ||||
|   hooks: | ||||
|   - id: editorconfig-checker | ||||
|     args: ['-disable-indentation'] | ||||
|     exclude: "^(.*\\.(bat)|LICENSE)$" | ||||
|  | ||||
| - repo: local | ||||
|   hooks: | ||||
|   - id: checkstyle | ||||
|     name: Check Java codestyle | ||||
|     files: ".*\\.java$" | ||||
|     language: system | ||||
|     entry: ./gradlew checkstyleMain checkstyleTest | ||||
|     pass_filenames: false | ||||
|     require_serial: true | ||||
|   - id: license | ||||
|     name: Check Java license headers | ||||
|     files: ".*\\.java$" | ||||
|     language: system | ||||
|     entry: ./gradlew licenseFormat | ||||
|     pass_filenames: false | ||||
|     require_serial: true | ||||
|   - id: illuaminate | ||||
|     name: Check Lua code | ||||
|     files: ".*\\.(lua|java|md)" | ||||
|     language: script | ||||
|     entry: config/pre-commit/illuaminate-lint.sh | ||||
|     pass_filenames: false | ||||
|     require_serial: true | ||||
|  | ||||
| exclude: | | ||||
|   (?x)^( | ||||
|     src/generated| | ||||
|     src/test/resources/test-rom/data/json-parsing/| | ||||
|     src/testMod/server-files/| | ||||
|     config/idea/| | ||||
|     .*\.dfpwm | ||||
|   ) | ||||
| @@ -1,16 +0,0 @@ | ||||
| #!/usr/bin/env sh | ||||
| set -e | ||||
|  | ||||
| test -d bin || mkdir bin | ||||
| test -f bin/illuaminate || curl -s -obin/illuaminate https://squiddev.cc/illuaminate/linux-x86-64/illuaminate | ||||
| chmod +x bin/illuaminate | ||||
|  | ||||
| if [ -n ${GITHUB_ACTIONS+x} ]; then | ||||
|     # Register a problem matcher (see https://github.com/actions/toolkit/blob/master/docs/problem-matchers.md) | ||||
|     # for illuaminate. | ||||
|     echo "::add-matcher::.github/matchers/illuaminate.json" | ||||
|     trap 'echo "::remove-matcher owner=illuaminate::"' EXIT | ||||
| fi | ||||
|  | ||||
| ./gradlew luaJavadoc | ||||
| bin/illuaminate lint | ||||
| @@ -1,21 +0,0 @@ | ||||
| --- | ||||
| module: [kind=event] alarm | ||||
| see: os.setAlarm To start an alarm. | ||||
| --- | ||||
|  | ||||
| The @{timer} event is fired when an alarm started with @{os.setAlarm} completes. | ||||
|  | ||||
| ## Return Values | ||||
| 1. @{string}: The event name. | ||||
| 2. @{number}: The ID of the alarm that finished. | ||||
|  | ||||
| ## Example | ||||
| Starts a timer and then prints its ID: | ||||
| ```lua | ||||
| local alarmID = os.setAlarm(os.time() + 0.05) | ||||
| local event, id | ||||
| repeat | ||||
|     event, id = os.pullEvent("alarm") | ||||
| until id == alarmID | ||||
| print("Alarm with ID " .. id .. " was fired") | ||||
| ``` | ||||
| @@ -1,24 +0,0 @@ | ||||
| --- | ||||
| module: [kind=event] char | ||||
| see: key To listen to any key press. | ||||
| --- | ||||
|  | ||||
| The @{char} event is fired when a character is _typed_ on the keyboard. | ||||
|  | ||||
| The @{char} event is different to a key press. Sometimes multiple key presses may result in one character being | ||||
| typed (for instance, on some European keyboards). Similarly, some keys (e.g. <kbd>Ctrl</kbd>) do not have any | ||||
| corresponding character. The @{key} should be used if you want to listen to key presses themselves. | ||||
|  | ||||
| ## Return values | ||||
| 1. @{string}: The event name. | ||||
| 2. @{string}: The string representing the character that was pressed. | ||||
|  | ||||
|  | ||||
| ## Example | ||||
| Prints each character the user presses: | ||||
| ```lua | ||||
| while true do | ||||
|   local event, character = os.pullEvent("char") | ||||
|   print(character .. " was pressed.") | ||||
| end | ||||
| ``` | ||||
| @@ -1,18 +0,0 @@ | ||||
| --- | ||||
| module: [kind=event] computer_command | ||||
| --- | ||||
|  | ||||
| The @{computer_command} event is fired when the `/computercraft queue` command is run for the current computer. | ||||
|  | ||||
| ## Return Values | ||||
| 1. @{string}: The event name. | ||||
| ... @{string}: The arguments passed to the command. | ||||
|  | ||||
| ## Example | ||||
| Prints the contents of messages sent: | ||||
| ```lua | ||||
| while true do | ||||
|   local event = {os.pullEvent("computer_command")} | ||||
|   print("Received message:", table.unpack(event, 2)) | ||||
| end | ||||
| ``` | ||||
| @@ -1,19 +0,0 @@ | ||||
| --- | ||||
| module: [kind=event] disk | ||||
| see: disk_eject For the event sent when a disk is removed. | ||||
| --- | ||||
|  | ||||
| The @{disk} event is fired when a disk is inserted into an adjacent or networked disk drive. | ||||
|  | ||||
| ## Return Values | ||||
| 1. @{string}: The event name. | ||||
| 2. @{string}: The side of the disk drive that had a disk inserted. | ||||
|  | ||||
| ## Example | ||||
| Prints a message when a disk is inserted: | ||||
| ```lua | ||||
| while true do | ||||
|   local event, side = os.pullEvent("disk") | ||||
|   print("Inserted a disk on side " .. side) | ||||
| end | ||||
| ``` | ||||
| @@ -1,19 +0,0 @@ | ||||
| --- | ||||
| module: [kind=event] disk_eject | ||||
| see: disk For the event sent when a disk is inserted. | ||||
| --- | ||||
|  | ||||
| The @{disk_eject} event is fired when a disk is removed from an adjacent or networked disk drive. | ||||
|  | ||||
| ## Return Values | ||||
| 1. @{string}: The event name. | ||||
| 2. @{string}: The side of the disk drive that had a disk removed. | ||||
|  | ||||
| ## Example | ||||
| Prints a message when a disk is removed: | ||||
| ```lua | ||||
| while true do | ||||
|   local event, side = os.pullEvent("disk_eject") | ||||
|   print("Removed a disk on side " .. side) | ||||
| end | ||||
| ``` | ||||
| @@ -1,14 +0,0 @@ | ||||
| --- | ||||
| module: [kind=event] http_check | ||||
| see: http.checkURLAsync To check a URL asynchronously. | ||||
| --- | ||||
|  | ||||
| The @{http_check} event is fired when a URL check finishes. | ||||
|  | ||||
| This event is normally handled inside @{http.checkURL}, but it can still be seen when using @{http.checkURLAsync}. | ||||
|  | ||||
| ## Return Values | ||||
| 1. @{string}: The event name. | ||||
| 2. @{string}: The URL requested to be checked. | ||||
| 3. @{boolean}: Whether the check succeeded. | ||||
| 4. @{string|nil}: If the check failed, a reason explaining why the check failed. | ||||
| @@ -1,39 +0,0 @@ | ||||
| --- | ||||
| module: [kind=event] http_failure | ||||
| see: http.request To send an HTTP request. | ||||
| --- | ||||
|  | ||||
| The @{http_failure} event is fired when an HTTP request fails. | ||||
|  | ||||
| This event is normally handled inside @{http.get} and @{http.post}, but it can still be seen when using @{http.request}. | ||||
|  | ||||
| ## Return Values | ||||
| 1. @{string}: The event name. | ||||
| 2. @{string}: The URL of the site requested. | ||||
| 3. @{string}: An error describing the failure. | ||||
| 4. @{http.Response|nil}: A response handle if the connection succeeded, but the server's response indicated failure. | ||||
|  | ||||
| ## Example | ||||
| Prints an error why the website cannot be contacted: | ||||
| ```lua | ||||
| local myURL = "https://does.not.exist.tweaked.cc" | ||||
| http.request(myURL) | ||||
| local event, url, err | ||||
| repeat | ||||
|     event, url, err = os.pullEvent("http_failure") | ||||
| until url == myURL | ||||
| print("The URL " .. url .. " could not be reached: " .. err) | ||||
| ``` | ||||
|  | ||||
| Prints the contents of a webpage that does not exist: | ||||
| ```lua | ||||
| local myURL = "https://tweaked.cc/this/does/not/exist" | ||||
| http.request(myURL) | ||||
| local event, url, err, handle | ||||
| repeat | ||||
|     event, url, err, handle = os.pullEvent("http_failure") | ||||
| until url == myURL | ||||
| print("The URL " .. url .. " could not be reached: " .. err) | ||||
| print(handle.getResponseCode()) | ||||
| handle.close() | ||||
| ``` | ||||
| @@ -1,27 +0,0 @@ | ||||
| --- | ||||
| module: [kind=event] http_success | ||||
| see: http.request To make an HTTP request. | ||||
| --- | ||||
|  | ||||
| The @{http_success} event is fired when an HTTP request returns successfully. | ||||
|  | ||||
| This event is normally handled inside @{http.get} and @{http.post}, but it can still be seen when using @{http.request}. | ||||
|  | ||||
| ## Return Values | ||||
| 1. @{string}: The event name. | ||||
| 2. @{string}: The URL of the site requested. | ||||
| 3. @{http.Response}: The handle for the response text. | ||||
|  | ||||
| ## Example | ||||
| Prints the content of a website (this may fail if the request fails): | ||||
| ```lua | ||||
| local myURL = "https://tweaked.cc/" | ||||
| http.request(myURL) | ||||
| local event, url, handle | ||||
| repeat | ||||
|     event, url, handle = os.pullEvent("http_success") | ||||
| until url == myURL | ||||
| print("Contents of " .. url .. ":") | ||||
| print(handle.readAll()) | ||||
| handle.close() | ||||
| ``` | ||||
| @@ -1,26 +0,0 @@ | ||||
| --- | ||||
| module: [kind=event] key | ||||
| --- | ||||
|  | ||||
| This event is fired when any key is pressed while the terminal is focused. | ||||
|  | ||||
| This event returns a numerical "key code" (for instance, <kbd>F1</kbd> is 290). This value may vary between versions and | ||||
| so it is recommended to use the constants in the @{keys} API rather than hard coding numeric values. | ||||
|  | ||||
| If the button pressed represented a printable character, then the @{key} event will be followed immediately by a @{char} | ||||
| event. If you are consuming text input, use a @{char} event instead! | ||||
|  | ||||
| ## Return values | ||||
| 1. @{string}: The event name. | ||||
| 2. @{number}: The numerical key value of the key pressed. | ||||
| 3. @{boolean}: Whether the key event was generated while holding the key (@{true}), rather than pressing it the first time (@{false}). | ||||
|  | ||||
| ## Example | ||||
| Prints each key when the user presses it, and if the key is being held. | ||||
|  | ||||
| ```lua | ||||
| while true do | ||||
|   local event, key, is_held = os.pullEvent("key") | ||||
|   print(("%s held=%s"):format(keys.getName(key), is_held)) | ||||
| end | ||||
| ``` | ||||
| @@ -1,24 +0,0 @@ | ||||
| --- | ||||
| module: [kind=event] key_up | ||||
| see: keys For a lookup table of the given keys. | ||||
| --- | ||||
|  | ||||
| Fired whenever a key is released (or the terminal is closed while a key was being pressed). | ||||
|  | ||||
| This event returns a numerical "key code" (for instance, <kbd>F1</kbd> is 290). This value may vary between versions and | ||||
| so it is recommended to use the constants in the @{keys} API rather than hard coding numeric values. | ||||
|  | ||||
| ## Return values | ||||
| 1. @{string}: The event name. | ||||
| 2. @{number}: The numerical key value of the key pressed. | ||||
|  | ||||
| ## Example | ||||
| Prints each key released on the keyboard whenever a @{key_up} event is fired. | ||||
|  | ||||
| ```lua | ||||
| while true do | ||||
|   local event, key = os.pullEvent("key_up") | ||||
|   local name = keys.getName(key) or "unknown key" | ||||
|   print(name .. " was released.") | ||||
| end | ||||
| ``` | ||||
| @@ -1,26 +0,0 @@ | ||||
| --- | ||||
| module: [kind=event] modem_message | ||||
| --- | ||||
|  | ||||
| The @{modem_message} event is fired when a message is received on an open channel on any @{modem}. | ||||
|  | ||||
| ## Return Values | ||||
| 1. @{string}: The event name. | ||||
| 2. @{string}: The side of the modem that received the message. | ||||
| 3. @{number}: The channel that the message was sent on. | ||||
| 4. @{number}: The reply channel set by the sender. | ||||
| 5. @{any}: The message as sent by the sender. | ||||
| 6. @{number}: The distance between the sender and the receiver, in blocks. | ||||
|  | ||||
| ## Example | ||||
| Wraps a @{modem} peripheral, opens channel 0 for listening, and prints all received messages. | ||||
|  | ||||
| ```lua | ||||
| local modem = peripheral.find("modem") or error("No modem attached", 0) | ||||
| modem.open(0) | ||||
|  | ||||
| while true do | ||||
|   local event, side, channel, replyChannel, message, distance = os.pullEvent("modem_message") | ||||
|   print(("Message received on side %s on channel %d (reply to %d) from %f blocks away with message %s"):format(side, channel, replyChannel, distance, tostring(message))) | ||||
| end | ||||
| ``` | ||||
| @@ -1,18 +0,0 @@ | ||||
| --- | ||||
| module: [kind=event] monitor_resize | ||||
| --- | ||||
|  | ||||
| The @{monitor_resize} event is fired when an adjacent or networked monitor's size is changed. | ||||
|  | ||||
| ## Return Values | ||||
| 1. @{string}: The event name. | ||||
| 2. @{string}: The side or network ID of the monitor that resized. | ||||
|  | ||||
| ## Example | ||||
| Prints a message when a monitor is resized: | ||||
| ```lua | ||||
| while true do | ||||
|   local event, side = os.pullEvent("monitor_resize") | ||||
|   print("The monitor on side " .. side .. " was resized.") | ||||
| end | ||||
| ``` | ||||
| @@ -1,20 +0,0 @@ | ||||
| --- | ||||
| module: [kind=event] monitor_touch | ||||
| --- | ||||
|  | ||||
| The @{monitor_touch} event is fired when an adjacent or networked Advanced Monitor is right-clicked. | ||||
|  | ||||
| ## Return Values | ||||
| 1. @{string}: The event name. | ||||
| 2. @{string}: The side or network ID of the monitor that was touched. | ||||
| 3. @{number}: The X coordinate of the touch, in characters. | ||||
| 4. @{number}: The Y coordinate of the touch, in characters. | ||||
|  | ||||
| ## Example | ||||
| Prints a message when a monitor is touched: | ||||
| ```lua | ||||
| while true do | ||||
|   local event, side, x, y = os.pullEvent("monitor_touch") | ||||
|   print("The monitor on side " .. side .. " was touched at (" .. x .. ", " .. y .. ")") | ||||
| end | ||||
| ``` | ||||
| @@ -1,32 +0,0 @@ | ||||
| --- | ||||
| module: [kind=event] mouse_click | ||||
| --- | ||||
|  | ||||
| This event is fired when the terminal is clicked with a mouse. This event is only fired on advanced computers (including | ||||
| advanced turtles and pocket computers). | ||||
|  | ||||
| ## Return values | ||||
| 1. @{string}: The event name. | ||||
| 2. @{number}: The mouse button that was clicked. | ||||
| 3. @{number}: The X-coordinate of the click. | ||||
| 4. @{number}: The Y-coordinate of the click. | ||||
|  | ||||
| ## Mouse buttons | ||||
| Several mouse events (@{mouse_click}, @{mouse_up}, @{mouse_scroll}) contain a "mouse button" code. This takes a | ||||
| numerical value depending on which button on your mouse was last pressed when this event occurred. | ||||
|  | ||||
| | Button Code | Mouse Button  | | ||||
| |------------:|---------------| | ||||
| |           1 | Left button   | | ||||
| |           2 | Right button  | | ||||
| |           3 | Middle button | | ||||
|  | ||||
| ## Example | ||||
| Print the button and the coordinates whenever the mouse is clicked. | ||||
|  | ||||
| ```lua | ||||
| while true do | ||||
|   local event, button, x, y = os.pullEvent("mouse_click") | ||||
|   print(("The mouse button %s was pressed at %d, %d"):format(button, x, y)) | ||||
| end | ||||
| ``` | ||||
| @@ -1,22 +0,0 @@ | ||||
| --- | ||||
| module: [kind=event] mouse_drag | ||||
| see: mouse_click For when a mouse button is initially pressed. | ||||
| --- | ||||
|  | ||||
| This event is fired every time the mouse is moved while a mouse button is being held. | ||||
|  | ||||
| ## Return values | ||||
| 1. @{string}: The event name. | ||||
| 2. @{number}: The [mouse button](mouse_click.html#Mouse_buttons) that is being pressed. | ||||
| 3. @{number}: The X-coordinate of the mouse. | ||||
| 4. @{number}: The Y-coordinate of the mouse. | ||||
|  | ||||
| ## Example | ||||
| Print the button and the coordinates whenever the mouse is dragged. | ||||
|  | ||||
| ```lua | ||||
| while true do | ||||
|   local event, button, x, y = os.pullEvent("mouse_drag") | ||||
|   print(("The mouse button %s was dragged at %d, %d"):format(button, x, y)) | ||||
| end | ||||
| ``` | ||||
| @@ -1,21 +0,0 @@ | ||||
| --- | ||||
| module: [kind=event] mouse_scroll | ||||
| --- | ||||
|  | ||||
| This event is fired when a mouse wheel is scrolled in the terminal. | ||||
|  | ||||
| ## Return values | ||||
| 1. @{string}: The event name. | ||||
| 2. @{number}: The direction of the scroll. (-1 = up, 1 = down) | ||||
| 3. @{number}: The X-coordinate of the mouse when scrolling. | ||||
| 4. @{number}: The Y-coordinate of the mouse when scrolling. | ||||
|  | ||||
| ## Example | ||||
| Prints the direction of each scroll, and the position of the mouse at the time. | ||||
|  | ||||
| ```lua | ||||
| while true do | ||||
|   local event, dir, x, y = os.pullEvent("mouse_scroll") | ||||
|   print(("The mouse was scrolled in direction %s at %d, %d"):format(dir, x, y)) | ||||
| end | ||||
| ``` | ||||
| @@ -1,21 +0,0 @@ | ||||
| --- | ||||
| module: [kind=event] mouse_up | ||||
| --- | ||||
|  | ||||
| This event is fired when a mouse button is released or a held mouse leaves the computer's terminal. | ||||
|  | ||||
| ## Return values | ||||
| 1. @{string}: The event name. | ||||
| 2. @{number}: The [mouse button](mouse_click.html#Mouse_buttons) that was released. | ||||
| 3. @{number}: The X-coordinate of the mouse. | ||||
| 4. @{number}: The Y-coordinate of the mouse. | ||||
|  | ||||
| ## Example | ||||
| Prints the coordinates and button number whenever the mouse is released. | ||||
|  | ||||
| ```lua | ||||
| while true do | ||||
|   local event, button, x, y = os.pullEvent("mouse_up") | ||||
|   print(("The mouse button %s was released at %d, %d"):format(button, x, y)) | ||||
| end | ||||
| ``` | ||||
| @@ -1,18 +0,0 @@ | ||||
| --- | ||||
| module: [kind=event] paste | ||||
| --- | ||||
|  | ||||
| The @{paste} event is fired when text is pasted into the computer through Ctrl-V (or ⌘V on Mac). | ||||
|  | ||||
| ## Return values | ||||
| 1. @{string}: The event name. | ||||
| 2. @{string} The text that was pasted. | ||||
|  | ||||
| ## Example | ||||
| Prints pasted text: | ||||
| ```lua | ||||
| while true do | ||||
|   local event, text = os.pullEvent("paste") | ||||
|   print('"' .. text .. '" was pasted') | ||||
| end | ||||
| ``` | ||||
| @@ -1,19 +0,0 @@ | ||||
| --- | ||||
| module: [kind=event] peripheral | ||||
| see: peripheral_detach For the event fired when a peripheral is detached. | ||||
| --- | ||||
|  | ||||
| The @{peripheral} event is fired when a peripheral is attached on a side or to a modem. | ||||
|  | ||||
| ## Return Values | ||||
| 1. @{string}: The event name. | ||||
| 2. @{string}: The side the peripheral was attached to. | ||||
|  | ||||
| ## Example | ||||
| Prints a message when a peripheral is attached: | ||||
| ```lua | ||||
| while true do | ||||
|   local event, side = os.pullEvent("peripheral") | ||||
|   print("A peripheral was attached on side " .. side) | ||||
| end | ||||
| ``` | ||||
| @@ -1,19 +0,0 @@ | ||||
| --- | ||||
| module: [kind=event] peripheral_detach | ||||
| see: peripheral For the event fired when a peripheral is attached. | ||||
| --- | ||||
|  | ||||
| The @{peripheral_detach} event is fired when a peripheral is detached from a side or from a modem. | ||||
|  | ||||
| ## Return Values | ||||
| 1. @{string}: The event name. | ||||
| 2. @{string}: The side the peripheral was detached from. | ||||
|  | ||||
| ## Example | ||||
| Prints a message when a peripheral is detached: | ||||
| ```lua | ||||
| while true do | ||||
|   local event, side = os.pullEvent("peripheral_detach") | ||||
|   print("A peripheral was detached on side " .. side) | ||||
| end | ||||
| ``` | ||||
| @@ -1,30 +0,0 @@ | ||||
| --- | ||||
| module: [kind=event] rednet_message | ||||
| see: modem_message For raw modem messages sent outside of Rednet. | ||||
| see: rednet.receive To wait for a Rednet message with an optional timeout and protocol filter. | ||||
| --- | ||||
|  | ||||
| The @{rednet_message} event is fired when a message is sent over Rednet. | ||||
|  | ||||
| This event is usually handled by @{rednet.receive}, but it can also be pulled manually. | ||||
|  | ||||
| @{rednet_message} events are sent by @{rednet.run} in the top-level coroutine in response to @{modem_message} events. A @{rednet_message} event is always preceded by a @{modem_message} event. They are generated inside CraftOS rather than being sent by the ComputerCraft machine. | ||||
|  | ||||
| ## Return Values | ||||
| 1. @{string}: The event name. | ||||
| 2. @{number}: The ID of the sending computer. | ||||
| 3. @{any}: The message sent. | ||||
| 4. @{string|nil}: The protocol of the message, if provided. | ||||
|  | ||||
| ## Example | ||||
| Prints a message when one is sent: | ||||
| ```lua | ||||
| while true do | ||||
|   local event, sender, message, protocol = os.pullEvent("rednet_message") | ||||
|   if protocol ~= nil then | ||||
|     print("Received message from " .. sender .. " with protocol " .. protocol .. " and message " .. tostring(message)) | ||||
|   else | ||||
|     print("Received message from " .. sender .. " with message " .. tostring(message)) | ||||
|   end | ||||
| end | ||||
| ``` | ||||
| @@ -1,14 +0,0 @@ | ||||
| --- | ||||
| module: [kind=event] redstone | ||||
| --- | ||||
|  | ||||
| The @{event!redstone} event is fired whenever any redstone inputs on the computer change. | ||||
|  | ||||
| ## Example | ||||
| Prints a message when a redstone input changes: | ||||
| ```lua | ||||
| while true do | ||||
|   os.pullEvent("redstone") | ||||
|   print("A redstone input has changed!") | ||||
| end | ||||
| ``` | ||||
| @@ -1,27 +0,0 @@ | ||||
| --- | ||||
| module: [kind=event] speaker_audio_empty | ||||
| see: speaker.playAudio To play audio using the speaker | ||||
| --- | ||||
|  | ||||
| ## Return Values | ||||
| 1. @{string}: The event name. | ||||
| 2. @{string}: The name of the speaker which is available to play more audio. | ||||
|  | ||||
|  | ||||
| ## Example | ||||
| This uses @{io.lines} to read audio data in blocks of 16KiB from "example_song.dfpwm", and then attempts to play it | ||||
| using @{speaker.playAudio}. If the speaker's buffer is full, it waits for an event and tries again. | ||||
|  | ||||
| ```lua {data-peripheral=speaker} | ||||
| local dfpwm = require("cc.audio.dfpwm") | ||||
| local speaker = peripheral.find("speaker") | ||||
|  | ||||
| local decoder = dfpwm.make_decoder() | ||||
| for chunk in io.lines("data/example.dfpwm", 16 * 1024) do | ||||
|     local buffer = decoder(chunk) | ||||
|  | ||||
|     while not speaker.playAudio(buffer) do | ||||
|         os.pullEvent("speaker_audio_empty") | ||||
|     end | ||||
| end | ||||
| ``` | ||||
| @@ -1,28 +0,0 @@ | ||||
| --- | ||||
| module: [kind=event] task_complete | ||||
| see: commands.execAsync To run a command which fires a task_complete event. | ||||
| --- | ||||
|  | ||||
| The @{task_complete} event is fired when an asynchronous task completes. This is usually handled inside the function call that queued the task; however, functions such as @{commands.execAsync} return immediately so the user can wait for completion. | ||||
|  | ||||
| ## Return Values | ||||
| 1. @{string}: The event name. | ||||
| 2. @{number}: The ID of the task that completed. | ||||
| 3. @{boolean}: Whether the command succeeded. | ||||
| 4. @{string}: If the command failed, an error message explaining the failure. (This is not present if the command succeeded.) | ||||
| ...: Any parameters returned from the command. | ||||
|  | ||||
| ## Example | ||||
| Prints the results of an asynchronous command: | ||||
| ```lua | ||||
| local taskID = commands.execAsync("say Hello") | ||||
| local event | ||||
| repeat | ||||
|     event = {os.pullEvent("task_complete")} | ||||
| until event[2] == taskID | ||||
| if event[3] == true then | ||||
|   print("Task " .. event[2] .. " succeeded:", table.unpack(event, 4)) | ||||
| else | ||||
|   print("Task " .. event[2] .. " failed: " .. event[4]) | ||||
| end | ||||
| ``` | ||||
| @@ -1,20 +0,0 @@ | ||||
| --- | ||||
| module: [kind=event] term_resize | ||||
| --- | ||||
|  | ||||
| The @{term_resize} event is fired when the main terminal is resized. For instance: | ||||
|  - When a the tab bar is shown or hidden in @{multishell}. | ||||
|  - When the terminal is redirected to a monitor via the "monitor" program and the monitor is resized. | ||||
|  | ||||
| When this event fires, some parts of the terminal may have been moved or deleted. Simple terminal programs (those | ||||
| not using @{term.setCursorPos}) can ignore this event, but more complex GUI programs should redraw the entire screen. | ||||
|  | ||||
| ## Example | ||||
| Prints : | ||||
| ```lua | ||||
| while true do | ||||
|   os.pullEvent("term_resize") | ||||
|   local w, h = term.getSize() | ||||
|   print("The term was resized to (" .. w .. ", " .. h .. ")") | ||||
| end | ||||
| ``` | ||||
| @@ -1,25 +0,0 @@ | ||||
| --- | ||||
| module: [kind=event] terminate | ||||
| --- | ||||
|  | ||||
| The @{terminate} event is fired when <kbd>Ctrl-T</kbd> is held down. | ||||
|  | ||||
| This event is normally handled by @{os.pullEvent}, and will not be returned. However, @{os.pullEventRaw} will return this event when fired. | ||||
|  | ||||
| @{terminate} will be sent even when a filter is provided to @{os.pullEventRaw}. When using @{os.pullEventRaw} with a filter, make sure to check that the event is not @{terminate}. | ||||
|  | ||||
| ## Example | ||||
| Prints a message when Ctrl-T is held: | ||||
| ```lua | ||||
| while true do | ||||
|   local event = os.pullEventRaw("terminate") | ||||
|   if event == "terminate" then print("Terminate requested!") end | ||||
| end | ||||
| ``` | ||||
|  | ||||
| Exits when Ctrl-T is held: | ||||
| ```lua | ||||
| while true do | ||||
|   os.pullEvent() | ||||
| end | ||||
| ``` | ||||
| @@ -1,21 +0,0 @@ | ||||
| --- | ||||
| module: [kind=event] timer | ||||
| see: os.startTimer To start a timer. | ||||
| --- | ||||
|  | ||||
| The @{timer} event is fired when a timer started with @{os.startTimer} completes. | ||||
|  | ||||
| ## Return Values | ||||
| 1. @{string}: The event name. | ||||
| 2. @{number}: The ID of the timer that finished. | ||||
|  | ||||
| ## Example | ||||
| Starts a timer and then prints its ID: | ||||
| ```lua | ||||
| local timerID = os.startTimer(2) | ||||
| local event, id | ||||
| repeat | ||||
|     event, id = os.pullEvent("timer") | ||||
| until id == timerID | ||||
| print("Timer with ID " .. id .. " was fired") | ||||
| ``` | ||||
| @@ -1,14 +0,0 @@ | ||||
| --- | ||||
| module: [kind=event] turtle_inventory | ||||
| --- | ||||
|  | ||||
| The @{turtle_inventory} event is fired when a turtle's inventory is changed. | ||||
|  | ||||
| ## Example | ||||
| Prints a message when the inventory is changed: | ||||
| ```lua | ||||
| while true do | ||||
|   os.pullEvent("turtle_inventory") | ||||
|   print("The inventory was changed.") | ||||
| end | ||||
| ``` | ||||
| @@ -1,21 +0,0 @@ | ||||
| --- | ||||
| module: [kind=event] websocket_closed | ||||
| --- | ||||
|  | ||||
| The @{websocket_closed} event is fired when an open WebSocket connection is closed. | ||||
|  | ||||
| ## Return Values | ||||
| 1. @{string}: The event name. | ||||
| 2. @{string}: The URL of the WebSocket that was closed. | ||||
|  | ||||
| ## Example | ||||
| Prints a message when a WebSocket is closed (this may take a minute): | ||||
| ```lua | ||||
| local myURL = "wss://example.tweaked.cc/echo" | ||||
| local ws = http.websocket(myURL) | ||||
| local event, url | ||||
| repeat | ||||
|     event, url = os.pullEvent("websocket_closed") | ||||
| until url == myURL | ||||
| print("The WebSocket at " .. url .. " was closed.") | ||||
| ``` | ||||
| @@ -1,25 +0,0 @@ | ||||
| --- | ||||
| module: [kind=event] websocket_failure | ||||
| see: http.websocketAsync To send an HTTP request. | ||||
| --- | ||||
|  | ||||
| The @{websocket_failure} event is fired when a WebSocket connection request fails. | ||||
|  | ||||
| This event is normally handled inside @{http.websocket}, but it can still be seen when using @{http.websocketAsync}. | ||||
|  | ||||
| ## Return Values | ||||
| 1. @{string}: The event name. | ||||
| 2. @{string}: The URL of the site requested. | ||||
| 3. @{string}: An error describing the failure. | ||||
|  | ||||
| ## Example | ||||
| Prints an error why the website cannot be contacted: | ||||
| ```lua | ||||
| local myURL = "wss://example.tweaked.cc/not-a-websocket" | ||||
| http.websocketAsync(myURL) | ||||
| local event, url, err | ||||
| repeat | ||||
|     event, url, err = os.pullEvent("websocket_failure") | ||||
| until url == myURL | ||||
| print("The URL " .. url .. " could not be reached: " .. err) | ||||
| ``` | ||||
| @@ -1,27 +0,0 @@ | ||||
| --- | ||||
| module: [kind=event] websocket_message | ||||
| --- | ||||
|  | ||||
| The @{websocket_message} event is fired when a message is received on an open WebSocket connection. | ||||
|  | ||||
| This event is normally handled by @{http.Websocket.receive}, but it can also be pulled manually. | ||||
|  | ||||
| ## Return Values | ||||
| 1. @{string}: The event name. | ||||
| 2. @{string}: The URL of the WebSocket. | ||||
| 3. @{string}: The contents of the message. | ||||
| 4. @{boolean}: Whether this is a binary message. | ||||
|  | ||||
| ## Example | ||||
| Prints a message sent by a WebSocket: | ||||
| ```lua | ||||
| local myURL = "wss://example.tweaked.cc/echo" | ||||
| local ws = http.websocket(myURL) | ||||
| ws.send("Hello!") | ||||
| local event, url, message | ||||
| repeat | ||||
|     event, url, message = os.pullEvent("websocket_message") | ||||
| until url == myURL | ||||
| print("Received message from " .. url .. " with contents " .. message) | ||||
| ws.close() | ||||
| ``` | ||||
| @@ -1,28 +0,0 @@ | ||||
| --- | ||||
| module: [kind=event] websocket_success | ||||
| see: http.websocketAsync To open a WebSocket asynchronously. | ||||
| --- | ||||
|  | ||||
| The @{websocket_success} event is fired when a WebSocket connection request returns successfully. | ||||
|  | ||||
| This event is normally handled inside @{http.websocket}, but it can still be seen when using @{http.websocketAsync}. | ||||
|  | ||||
| ## Return Values | ||||
| 1. @{string}: The event name. | ||||
| 2. @{string}: The URL of the site. | ||||
| 3. @{http.Websocket}: The handle for the WebSocket. | ||||
|  | ||||
| ## Example | ||||
| Prints the content of a website (this may fail if the request fails): | ||||
| ```lua | ||||
| local myURL = "wss://example.tweaked.cc/echo" | ||||
| http.websocketAsync(myURL) | ||||
| local event, url, handle | ||||
| repeat | ||||
|     event, url, handle = os.pullEvent("websocket_success") | ||||
| until url == myURL | ||||
| print("Connected to " .. url) | ||||
| handle.send("Hello!") | ||||
| print(handle.receive()) | ||||
| handle.close() | ||||
| ``` | ||||
| @@ -1,99 +0,0 @@ | ||||
| --- | ||||
| module: [kind=guide] local_ips | ||||
| --- | ||||
|  | ||||
| # Allowing access to local IPs | ||||
| By default, ComputerCraft blocks access to local IP addresses for security. This means you can't normally access any | ||||
| HTTP server running on your computer. However, this may be useful for testing programs without having a remote | ||||
| server. You can unblock these IPs in the ComputerCraft config. | ||||
|  | ||||
|  - [Minecraft 1.13 and later, CC:T 1.87.0 and later](#cc-1.87.0) | ||||
|  - [Minecraft 1.13 and later, CC:T 1.86.2 and earlier](#cc-1.86.2) | ||||
|  - [Minecraft 1.12.2 and earlier](#mc-1.12) | ||||
|  | ||||
| ## Minecraft 1.13 and later, CC:T 1.87.0 and later {#cc-1.87.0} | ||||
| The configuration file can be located at `serverconfig/computercraft-server.toml` inside the world folder on either | ||||
| single-player or multiplayer. Look for lines that look like this: | ||||
|  | ||||
| ```toml | ||||
| #A list of rules which control behaviour of the "http" API for specific domains or IPs. | ||||
| #Each rule is an item with a 'host' to match against, and a series of properties. The host may be a domain name ("pastebin.com"), | ||||
| #wildcard ("*.pastebin.com") or CIDR notation ("127.0.0.0/8"). If no rules, the domain is blocked. | ||||
| [[http.rules]] | ||||
|     host = "$private" | ||||
|     action = "deny" | ||||
| ``` | ||||
|  | ||||
| On 1.95.0 and later, this will be a single entry with `host = "$private"`. On earlier versions, this will be a number of | ||||
| `[[http.rules]]` with various IP addresses. You will want to remove all of the `[[http.rules]]` entires that have | ||||
| `action = "deny"`. Then save the file and relaunch Minecraft (Server). | ||||
|  | ||||
| Here's what it should look like after removing: | ||||
|  | ||||
| ```toml | ||||
| #A list of rules which control behaviour of the "http" API for specific domains or IPs. | ||||
| #Each rule is an item with a 'host' to match against, and a series of properties. The host may be a domain name ("pastebin.com"), | ||||
| #wildcard ("*.pastebin.com") or CIDR notation ("127.0.0.0/8"). If no rules, the domain is blocked. | ||||
| [[http.rules]] | ||||
|     #The maximum size (in bytes) that a computer can send or receive in one websocket packet. | ||||
|     max_websocket_message = 131072 | ||||
|     host = "*" | ||||
|     #The maximum size (in bytes) that a computer can upload in a single request. This includes headers and POST text. | ||||
|     max_upload = 4194304 | ||||
|     action = "allow" | ||||
|     #The maximum size (in bytes) that a computer can download in a single request. Note that responses may receive more data than allowed, but this data will not be returned to the client. | ||||
|     max_download = 16777216 | ||||
|     #The period of time (in milliseconds) to wait before a HTTP request times out. Set to 0 for unlimited. | ||||
|     timeout = 30000 | ||||
| ``` | ||||
|  | ||||
| ## Minecraft 1.13 and later, CC:T 1.86.2 and earlier {#cc-1.86.2} | ||||
| The configuration file for singleplayer is at `.minecraft/config/computercraft-common.toml`. Look for lines that look | ||||
| like this: | ||||
|  | ||||
| ```toml | ||||
| #A list of wildcards for domains or IP ranges that cannot be accessed through the "http" API on Computers. | ||||
| #If this is empty then all whitelisted domains will be accessible. Example: "*.github.com" will block access to all subdomains of github.com. | ||||
| #You can use domain names ("pastebin.com"), wilcards ("*.pastebin.com") or CIDR notation ("127.0.0.0/8"). | ||||
| blacklist = ["127.0.0.0/8", "10.0.0.0/8", "172.16.0.0/12", "192.168.0.0/16", "fd00::/8"] | ||||
| ``` | ||||
|  | ||||
| Remove everything inside the array, leaving the last line as `blacklist = []`. Then save the file and relaunch Minecraft. | ||||
|  | ||||
| Here's what it should look like after removing: | ||||
|  | ||||
| ```toml | ||||
| #A list of wildcards for domains or IP ranges that cannot be accessed through the "http" API on Computers. | ||||
| #If this is empty then all whitelisted domains will be accessible. Example: "*.github.com" will block access to all subdomains of github.com. | ||||
| #You can use domain names ("pastebin.com"), wilcards ("*.pastebin.com") or CIDR notation ("127.0.0.0/8"). | ||||
| blacklist = [] | ||||
| ``` | ||||
|  | ||||
| ## Minecraft 1.12.2 and earlier {#mc-1.12} | ||||
| On singleplayer, the configuration file is located at `.minecraft\config\ComputerCraft.cfg`. On multiplayer, the | ||||
| configuration file is located at `<server folder>\config\ComputerCraft.cfg`. Look for lines that look like this: | ||||
|  | ||||
| ```ini | ||||
| # A list of wildcards for domains or IP ranges that cannot be accessed through the "http" API on Computers. | ||||
| # If this is empty then all explicitly allowed domains will be accessible. Example: "*.github.com" will block access to all subdomains of github.com. | ||||
| # You can use domain names ("pastebin.com"), wildcards ("*.pastebin.com") or CIDR notation ("127.0.0.0/8"). | ||||
| S:blocked_domains < | ||||
|     127.0.0.0/8 | ||||
|     10.0.0.0/8 | ||||
|     172.16.0.0/12 | ||||
|     192.168.0.0/16 | ||||
|     fd00::/8 | ||||
|  > | ||||
| ``` | ||||
|  | ||||
| Delete everything between the `<>`, leaving the last line as `S:blocked_domains = <>`. Then save the file and relaunch | ||||
| Minecraft (Server). | ||||
|  | ||||
| Here's what it should look like after removing: | ||||
|  | ||||
| ```ini | ||||
| # A list of wildcards for domains or IP ranges that cannot be accessed through the "http" API on Computers. | ||||
| # If this is empty then all explicitly allowed domains will be accessible. Example: "*.github.com" will block access to all subdomains of github.com. | ||||
| # You can use domain names ("pastebin.com"), wildcards ("*.pastebin.com") or CIDR notation ("127.0.0.0/8"). | ||||
| S:blocked_domains <> | ||||
| ``` | ||||
| @@ -1,205 +0,0 @@ | ||||
| --- | ||||
| module: [kind=guide] speaker_audio | ||||
| see: speaker.playAudio Play PCM audio using a speaker. | ||||
| see: cc.audio.dfpwm Provides utilities for encoding and decoding DFPWM files. | ||||
| --- | ||||
|  | ||||
| # Playing audio with speakers | ||||
| CC: Tweaked's speaker peripheral provides a powerful way to play any audio you like with the @{speaker.playAudio} | ||||
| method. However, for people unfamiliar with digital audio, it's not the most intuitive thing to use. This guide provides | ||||
| an introduction to digital audio, demonstrates how to play music with CC: Tweaked's speakers, and then briefly discusses | ||||
| the more complex topic of audio processing. | ||||
|  | ||||
| ## A short introduction to digital audio | ||||
| When sound is recorded it is captured as an analogue signal, effectively the electrical version of a sound | ||||
| wave. However, this signal is continuous, and so can't be used directly by a computer. Instead, we measure (or *sample*) | ||||
| the amplitude of the wave many times a second and then *quantise* that amplitude, rounding it to the nearest | ||||
| representable value. | ||||
|  | ||||
| This representation of sound - a long, uniformally sampled list of amplitudes is referred to as [Pulse-code | ||||
| Modulation][PCM] (PCM). PCM can be thought of as the "standard" audio format, as it's incredibly easy to work with. For | ||||
| instance, to mix two pieces of audio together, you can just add samples from the two tracks together and take the average. | ||||
|  | ||||
| CC: Tweaked's speakers also work with PCM audio. It plays back 48,000 samples a second, where each sample is an integer | ||||
| between -128 and 127. This is more commonly referred to as 48kHz and an 8-bit resolution. | ||||
|  | ||||
| Let's now look at a quick example. We're going to generate a [Sine Wave] at 220Hz, which sounds like a low monotonous | ||||
| hum. First we wrap our speaker peripheral, and then we fill a table (also referred to as a *buffer*) with 128×1024 | ||||
| samples - this is the maximum number of samples a speaker can accept in one go. | ||||
|  | ||||
| In order to fill this buffer, we need to do a little maths. We want to play 220 sine waves each second, where each sine | ||||
| wave completes a full oscillation in 2π "units". This means one seconds worth of audio is 2×π×220 "units" long. We then | ||||
| need to split this into 48k samples, basically meaning for each sample we move 2×π×220/48k "along" the sine curve. | ||||
|  | ||||
| ```lua {data-peripheral=speaker} | ||||
| local speaker = peripheral.find("speaker") | ||||
|  | ||||
| local buffer = {} | ||||
| local t, dt = 0, 2 * math.pi * 220 / 48000 | ||||
| for i = 1, 128 * 1024 do | ||||
|     buffer[i] = math.floor(math.sin(t) * 127) | ||||
|     t = (t + dt) % (math.pi * 2) | ||||
| end | ||||
|  | ||||
| speaker.playAudio(buffer) | ||||
| ``` | ||||
|  | ||||
| ## Streaming audio | ||||
| You might notice that the above snippet only generates a short bit of audio - 2.7s seconds to be precise. While we could | ||||
| try increasing the number of loop iterations, we'll get an error when we try to play it through the speaker: the sound | ||||
| buffer is too large for it to handle. | ||||
|  | ||||
| Our 2.7 seconds of audio is stored in a table with over 130 _thousand_ elements. If we wanted to play a full minute of | ||||
| sine waves (and why wouldn't you?), you'd need a table with almost 3 _million_. Suddenly you find these numbers adding | ||||
| up very quickly, and these tables take up more and more memory. | ||||
|  | ||||
| Instead of building our entire song (well, sine wave) in one go, we can produce it in small batches, each of which get | ||||
| passed off to @{speaker.playAudio} when the time is right. This allows us to build a _stream_ of audio, where we read | ||||
| chunks of audio one at a time (either from a file or a tone generator like above), do some optional processing to each | ||||
| one, and then play them. | ||||
|  | ||||
| Let's adapt our example from above to do that instead. | ||||
|  | ||||
| ```lua {data-peripheral=speaker} | ||||
| local speaker = peripheral.find("speaker") | ||||
|  | ||||
| local t, dt = 0, 2 * math.pi * 220 / 48000 | ||||
| while true do | ||||
|     local buffer = {} | ||||
|     for i = 1, 16 * 1024 * 8 do | ||||
|         buffer[i] = math.floor(math.sin(t) * 127) | ||||
|         t = (t + dt) % (math.pi * 2) | ||||
|     end | ||||
|  | ||||
|     while not speaker.playAudio(buffer) do | ||||
|         os.pullEvent("speaker_audio_empty") | ||||
|     end | ||||
| end | ||||
| ``` | ||||
|  | ||||
| It looks pretty similar to before, aside from we've wrapped the generation and playing code in a while loop, and added a | ||||
| rather odd loop with @{speaker.playAudio} and @{os.pullEvent}. | ||||
|  | ||||
| Let's talk about this loop, why do we need to keep calling @{speaker.playAudio}? Remember that what we're trying to do | ||||
| here is avoid keeping too much audio in memory at once. However, if we're generating audio quicker than the speakers can | ||||
| play it, we're not helping at all - all this audio is still hanging around waiting to be played! | ||||
|  | ||||
| In order to avoid this, the speaker rejects any new chunks of audio if its backlog is too large. When this happens, | ||||
| @{speaker.playAudio} returns false. Once enough audio has played, and the backlog has been reduced, a | ||||
| @{speaker_audio_empty} event is queued, and we can try to play our chunk once more. | ||||
|  | ||||
| ## Storing audio | ||||
| PCM is a fantastic way of representing audio when we want to manipulate it, but it's not very efficient when we want to | ||||
| store it to disk. Compare the size of a WAV file (which uses PCM) to an equivalent MP3, it's often 5 times the size. | ||||
| Instead, we store audio in special formats (or *codecs*) and then convert them to PCM when we need to do processing on | ||||
| them. | ||||
|  | ||||
| Modern audio codecs use some incredibly impressive techniques to compress the audio as much as possible while preserving | ||||
| sound quality. However, due to CC: Tweaked's limited processing power, it's not really possible to use these from your | ||||
| computer. Instead, we need something much simpler. | ||||
|  | ||||
| DFPWM (Dynamic Filter Pulse Width Modulation) is the de facto standard audio format of the ComputerCraft (and | ||||
| OpenComputers) world. Originally popularised by the addon mod [Computronics], CC:T now has built-in support for it with | ||||
| the @{cc.audio.dfpwm} module. This allows you to read DFPWM files from disk, decode them to PCM, and then play them | ||||
| using the speaker. | ||||
|  | ||||
| Let's dive in with an example, and we'll explain things afterwards: | ||||
|  | ||||
| ```lua {data-peripheral=speaker} | ||||
| local dfpwm = require("cc.audio.dfpwm") | ||||
| local speaker = peripheral.find("speaker") | ||||
|  | ||||
| local decoder = dfpwm.make_decoder() | ||||
| for chunk in io.lines("data/example.dfpwm", 16 * 1024) do | ||||
|     local buffer = decoder(chunk) | ||||
|  | ||||
|     while not speaker.playAudio(buffer) do | ||||
|         os.pullEvent("speaker_audio_empty") | ||||
|     end | ||||
| end | ||||
| ``` | ||||
|  | ||||
| Once again, we see the @{speaker.playAudio}/@{speaker_audio_empty} loop. However, the rest of the program is a little | ||||
| different. | ||||
|  | ||||
| First, we require the dfpwm module and call @{cc.audio.dfpwm.make_decoder} to construct a new decoder. This decoder | ||||
| accepts blocks of DFPWM data and converts it to a list of 8-bit amplitudes, which we can then play with our speaker. | ||||
|  | ||||
| As mentioned above, @{speaker.playAudio} accepts at most 128×1024 samples in one go. DFPMW uses a single bit for each | ||||
| sample, which means we want to process our audio in chunks of 16×1024 bytes (16KiB). In order to do this, we use | ||||
| @{io.lines}, which provides a nice way to loop over chunks of a file. You can of course just use @{fs.open} and | ||||
| @{fs.BinaryReadHandle.read} if you prefer. | ||||
|  | ||||
| ## Processing audio | ||||
| As mentioned near the beginning of this guide, PCM audio is pretty easy to work with as it's just a list of amplitudes. | ||||
| You can mix together samples from different streams by adding their amplitudes, change the rate of playback by removing | ||||
| samples, etc... | ||||
|  | ||||
| Let's put together a small demonstration here. We're going to add a small delay effect to the song above, so that you | ||||
| hear a faint echo a second and a half later. | ||||
|  | ||||
| In order to do this, we'll follow a format similar to the previous example, decoding the audio and then playing it. | ||||
| However, we'll also add some new logic between those two steps, which loops over every sample in our chunk of audio, and | ||||
| adds the sample from 1.5 seconds ago to it. | ||||
|  | ||||
| For this, we'll need to keep track of the last 72k samples - exactly 1.5 seconds worth of audio. We can do this using a | ||||
| [Ring Buffer], which helps makes things a little more efficient. | ||||
|  | ||||
| ```lua {data-peripheral=speaker} | ||||
| local dfpwm = require("cc.audio.dfpwm") | ||||
| local speaker = peripheral.find("speaker") | ||||
|  | ||||
| -- Speakers play at 48kHz, so 1.5 seconds is 72k samples. We first fill our buffer | ||||
| -- with 0s, as there's nothing to echo at the start of the track! | ||||
| local samples_i, samples_n = 1, 48000 * 1.5 | ||||
| local samples = {} | ||||
| for i = 1, samples_n do samples[i] = 0 end | ||||
|  | ||||
| local decoder = dfpwm.make_decoder() | ||||
| for chunk in io.lines("data/example.dfpwm", 16 * 1024) do | ||||
|     local buffer = decoder(chunk) | ||||
|  | ||||
|     for i = 1, #buffer do | ||||
|         local original_value = buffer[i] | ||||
|  | ||||
|         -- Replace this sample with its current amplitude plus the amplitude from 1.5 seconds ago. | ||||
|         -- We scale both to ensure the resulting value is still between -128 and 127. | ||||
|         buffer[i] = original_value * 0.6 + samples[samples_i] * 0.4 | ||||
|  | ||||
|         -- Now store the current sample, and move the "head" of our ring buffer forward one place. | ||||
|         samples[samples_i] = original_value | ||||
|         samples_i = samples_i + 1 | ||||
|         if samples_i > samples_n then samples_i = 1 end | ||||
|     end | ||||
|  | ||||
|     while not speaker.playAudio(buffer) do | ||||
|         os.pullEvent("speaker_audio_empty") | ||||
|     end | ||||
|  | ||||
|     -- The audio processing above can be quite slow and preparing the first batch of audio | ||||
|     -- may timeout the computer. We sleep to avoid this. | ||||
|     -- There's definitely better ways of handling this - this is just an example! | ||||
|     sleep(0.05) | ||||
| end | ||||
| ``` | ||||
|  | ||||
| :::note Confused? | ||||
| Don't worry if you don't understand this example. It's quite advanced, and does use some ideas that this guide doesn't | ||||
| cover. That said, don't be afraid to ask on [Discord] or [IRC] either! | ||||
| ::: | ||||
|  | ||||
| It's worth noting that the examples of audio processing we've mentioned here are about manipulating the _amplitude_ of | ||||
| the wave. If you wanted to modify the _frequency_ (for instance, shifting the pitch), things get rather more complex. | ||||
| For this, you'd need to use the [Fast Fourier transform][FFT] to convert the stream of amplitudes to frequencies, | ||||
| process those, and then convert them back to amplitudes. | ||||
|  | ||||
| This is, I'm afraid, left as an exercise to the reader. | ||||
|  | ||||
| [Computronics]: https://github.com/Vexatos/Computronics/ "Computronics on GitHub" | ||||
| [FFT]: https://en.wikipedia.org/wiki/Fast_Fourier_transform "Fast Fourier transform - Wikipedia" | ||||
| [PCM]: https://en.wikipedia.org/wiki/Pulse-code_modulation "Pulse-code Modulation - Wikipedia" | ||||
| [Ring Buffer]: https://en.wikipedia.org/wiki/Circular_buffer "Circular buffer - Wikipedia" | ||||
| [Sine Wave]: https://en.wikipedia.org/wiki/Sine_wave "Sine wave - Wikipedia" | ||||
|  | ||||
| [Discord]: https://discord.computercraft.cc "The Minecraft Computer Mods Discord" | ||||
| [IRC]: http://webchat.esper.net/?channels=computercraft "IRC webchat on EsperNet" | ||||
| @@ -1,83 +0,0 @@ | ||||
| --- | ||||
| module: [kind=guide] using_require | ||||
| --- | ||||
|  | ||||
| # Reusing code with require | ||||
| A library is a collection of useful functions and other definitions which is stored separately to your main program. You | ||||
| might want to create a library because you have some functions which are used in multiple programs, or just to split | ||||
| your program into multiple more modular files. | ||||
|  | ||||
| Let's say we want to create a small library to make working with the @{term|terminal} a little easier. We'll provide two | ||||
| functions: `reset`, which clears the terminal and sets the cursor to (1, 1), and `write_center`, which prints some text | ||||
| in the middle of the screen. | ||||
|  | ||||
| Start off by creating a file called `more_term.lua`: | ||||
|  | ||||
| ```lua {data-snippet=more_term} | ||||
| local function reset() | ||||
|   term.clear() | ||||
|   term.setCursorPos(1, 1) | ||||
| end | ||||
|  | ||||
| local function write_center(text) | ||||
|   local x, y = term.getCursorPos() | ||||
|   local width, height = term.getSize() | ||||
|   term.setCursorPos(math.floor((width - #text) / 2) + 1, y) | ||||
|   term.write(text) | ||||
| end | ||||
|  | ||||
| return { reset = reset, write_center = write_center } | ||||
| ``` | ||||
|  | ||||
| Now, what's going on here? We define our two functions as one might expect, and then at the bottom return a table with | ||||
| the two functions. When we require this library, this table is what is returned. With that, we can then call the | ||||
| original functions. Now create a new file, with the following: | ||||
|  | ||||
| ```lua {data-mount=more_term:more_term.lua} | ||||
| local more_term = require("more_term") | ||||
| more_term.reset() | ||||
| more_term.write_center("Hello, world!") | ||||
| ``` | ||||
|  | ||||
| When run, this'll clear the screen and print some text in the middle of the first line. | ||||
|  | ||||
| ## require in depth | ||||
| While the previous section is a good introduction to how @{require} operates, there are a couple of remaining points | ||||
| which are worth mentioning for more advanced usage. | ||||
|  | ||||
| ### Libraries can return anything | ||||
| In our above example, we return a table containing the functions we want to expose. However, it's worth pointing out | ||||
| that you can return ''anything'' from your library - a table, a function or even just a string! @{require} treats them | ||||
| all the same, and just returns whatever your library provides. | ||||
|  | ||||
| ### Module resolution and the package path | ||||
| In the above examples, we defined our library in a file, and @{require} read from it. While this is what you'll do most | ||||
| of the time, it is possible to make @{require} look elsewhere for your library, such as downloading from a website or | ||||
| loading from an in-memory library store. | ||||
|  | ||||
| As a result, the *module name* you pass to @{require} doesn't correspond to a file path. One common mistake is to load | ||||
| code from a sub-directory using `require("folder/library")` or even `require("folder/library.lua")`, neither of which | ||||
| will do quite what you expect. | ||||
|  | ||||
| When loading libraries (also referred to as *modules*) from files, @{require} searches along the *@{package.path|module | ||||
| path}*. By default, this looks something like: | ||||
|  | ||||
| * `?.lua` | ||||
| * `?/init.lua` | ||||
| * `/rom/modules/main/?.lua` | ||||
| * etc... | ||||
|  | ||||
| When you call `require("my_library")`, @{require} replaces the `?` in each element of the path with your module name, and | ||||
| checks if the file exists. In this case, we'd look for `my_library.lua`, `my_library/init.lua`, | ||||
| `/rom/modules/main/my_library.lua` and so on. Note that this works *relative to the current program*, so if your | ||||
| program is actually called `folder/program`, then we'll look for `folder/my_library.lua`, etc... | ||||
|  | ||||
| One other caveat is loading libraries from sub-directories. For instance, say we have a file | ||||
| `my/fancy/library.lua`. This can be loaded by using `require("my.fancy.library")` - the '.'s are replaced with '/' | ||||
| before we start looking for the library. | ||||
|  | ||||
| ## External links | ||||
| There are several external resources which go into require in a little more detail: | ||||
|  | ||||
|  - The [Lua Module tutorial](http://lua-users.org/wiki/ModulesTutorial) on the Lua wiki. | ||||
|  - [Lua's manual section on @{require}](https://www.lua.org/manual/5.1/manual.html#pdf-require). | ||||
| @@ -1 +0,0 @@ | ||||
| <meta name="theme-color" content="#c8d87c"> | ||||
| Before Width: | Height: | Size: 53 KiB | 
| Before Width: | Height: | Size: 194 KiB | 
| Before Width: | Height: | Size: 163 KiB | 
							
								
								
									
										55
									
								
								doc/index.md
									
									
									
									
									
								
							
							
						
						| @@ -1,55 +0,0 @@ | ||||
| #  | ||||
| CC: Tweaked is a mod for Minecraft which adds programmable computers, turtles and more to the game. A fork of the | ||||
| much-beloved [ComputerCraft], it continues its legacy with better performance, stability, and a wealth of new features. | ||||
|  | ||||
| CC: Tweaked can be installed from [CurseForge] or [Modrinth]. It requires the [Minecraft Forge][forge] mod loader, but | ||||
| [versions are available for Fabric][ccrestitched]. | ||||
|  | ||||
| ## Features | ||||
| Controlled using the [Lua programming language][lua], CC: Tweaked's computers provides all the tools you need to start | ||||
| writing code and automating your Minecraft world. | ||||
|  | ||||
| {.big-image} | ||||
|  | ||||
| While computers are incredibly powerful, they're rather limited by their inability to move about. *Turtles* are the | ||||
| solution here. They can move about the world, placing and breaking blocks, swinging a sword to protect you from zombies, | ||||
| or whatever else you program them to! | ||||
|  | ||||
| {.big-image} | ||||
|  | ||||
| Not all problems can be solved with a pickaxe though, and so CC: Tweaked also provides a bunch of additional peripherals | ||||
| for your computers. You can play a tune with speakers, display text or images on a monitor, connect all your | ||||
| computers together with modems, and much more. | ||||
|  | ||||
| Computers can now also interact with inventories such as chests, allowing you to build complex inventory and item | ||||
| management systems. | ||||
|  | ||||
| {.big-image} | ||||
|  | ||||
| ## Getting Started | ||||
| While ComputerCraft is lovely for both experienced programmers and for people who have never coded before, it can be a | ||||
| little daunting getting started. Thankfully, there's several fantastic tutorials out there: | ||||
|  | ||||
|  - [Direwolf20's ComputerCraft tutorials](https://www.youtube.com/watch?v=wrUHUhfCY5A "ComputerCraft Tutorial Episode 1 - HELP! and Hello World") | ||||
|  - [Sethbling's ComputerCraft series](https://www.youtube.com/watch?v=DSsx4VSe-Uk "Programming Tutorial with Minecraft Turtles -- Ep. 1: Intro to Turtles and If-Then-Else_End") | ||||
|  - [Lyqyd's Computer Basics 1](http://www.computercraft.info/forums2/index.php?/topic/15033-computer-basics-i/ "Computer Basics I") | ||||
|  | ||||
| Once you're a little more familiar with the mod, the sidebar and links below provide more detailed documentation on the | ||||
| various APIs and peripherals provided by the mod. | ||||
|  | ||||
| If you get stuck, do pop in to the [Minecraft Computer Mod Discord guild][discord] or ComputerCraft's | ||||
| [IRC channel][irc]. | ||||
|  | ||||
| ## Get Involved | ||||
| CC: Tweaked lives on [GitHub]. If you've got any ideas, feedback or bugs please do [create an issue][bug]. | ||||
|  | ||||
| [github]: https://github.com/cc-tweaked/CC-Tweaked/ "CC: Tweaked on GitHub" | ||||
| [bug]: https://github.com/cc-tweaked/CC-Tweaked/issues/new/choose | ||||
| [computercraft]: https://github.com/dan200/ComputerCraft "ComputerCraft on GitHub" | ||||
| [curseforge]: https://minecraft.curseforge.com/projects/cc-tweaked "Download CC: Tweaked from CurseForge" | ||||
| [modrinth]: https://modrinth.com/mod/gu7yAYhd "Download CC: Tweaked from Modrinth" | ||||
| [forge]: https://files.minecraftforge.net/ "Download Minecraft Forge." | ||||
| [ccrestitched]: https://www.curseforge.com/minecraft/mc-mods/cc-restitched "Download CC: Restitched from CurseForge" | ||||
| [lua]: https://www.lua.org/ "Lua's main website" | ||||
| [discord]: https://discord.computercraft.cc "The Minecraft Computer Mods Discord" | ||||
| [irc]: http://webchat.esper.net/?channels=computercraft "IRC webchat on EsperNet" | ||||
							
								
								
									
										
											BIN
										
									
								
								doc/logo.png
									
									
									
									
									
								
							
							
						
						| Before Width: | Height: | Size: 1.5 KiB | 
| @@ -1,95 +0,0 @@ | ||||
| --- | ||||
| module: [kind=reference] feature_compat | ||||
| --- | ||||
|  | ||||
| # Lua 5.2/5.3 features in CC: Tweaked | ||||
| CC: Tweaked is based off of the Cobalt Lua runtime, which uses Lua 5.1. However, Cobalt and CC:T implement additional features from Lua 5.2 and 5.3 (as well as some deprecated 5.0 features) that are not available in base 5.1. This page lists all of the compatibility for these newer versions. | ||||
|  | ||||
| ## Lua 5.2 | ||||
| | Feature                                                       | Supported? | Notes                                                             | | ||||
| |---------------------------------------------------------------|------------|-------------------------------------------------------------------| | ||||
| | `goto`/labels                                                 | ❌         |                                                                   | | ||||
| | `_ENV`                                                        | 🔶         | The `_ENV` global points to `getfenv()`, but it cannot be set.    | | ||||
| | `\z` escape                                                   | ✔          |                                                                   | | ||||
| | `\xNN` escape                                                 | ✔          |                                                                   | | ||||
| | Hex literal fractional/exponent parts                         | ✔          |                                                                   | | ||||
| | Empty statements                                              | ❌         |                                                                   | | ||||
| | `__len` metamethod                                            | ✔          |                                                                   | | ||||
| | `__ipairs` metamethod                                         | ❌         |                                                                   | | ||||
| | `__pairs` metamethod                                          | ✔          |                                                                   | | ||||
| | `bit32` library                                               | ✔          |                                                                   | | ||||
| | `collectgarbage` isrunning, generational, incremental options | ❌         | `collectgarbage` does not exist in CC:T.                          | | ||||
| | New `load` syntax                                             | ✔          |                                                                   | | ||||
| | `loadfile` mode parameter                                     | ✔          | Supports both 5.1 and 5.2+ syntax.                                | | ||||
| | Removed `loadstring`                                          | 🔶         | Only if `disable_lua51_features` is enabled in the configuration. | | ||||
| | Removed `getfenv`, `setfenv`                                  | 🔶         | Only if `disable_lua51_features` is enabled in the configuration. | | ||||
| | `rawlen` function                                             | ✔          |                                                                   | | ||||
| | Negative index to `select`                                    | ✔          |                                                                   | | ||||
| | Removed `unpack`                                              | 🔶         | Only if `disable_lua51_features` is enabled in the configuration. | | ||||
| | Arguments to `xpcall`                                         | ❌         |                                                                   | | ||||
| | Second return value from `coroutine.running`                  | ❌         |                                                                   | | ||||
| | Removed `module`                                              | ✔          |                                                                   | | ||||
| | `package.loaders` -> `package.searchers`                      | ❌         |                                                                   | | ||||
| | Second argument to loader functions                           | ✔          |                                                                   | | ||||
| | `package.config`                                              | ✔          |                                                                   | | ||||
| | `package.searchpath`                                          | ✔          |                                                                   | | ||||
| | Removed `package.seeall`                                      | ✔          |                                                                   | | ||||
| | `string.dump` on functions with upvalues (blanks them out)    | ✔          |                                                                   | | ||||
| | `string.rep` separator                                        | ❌         |                                                                   | | ||||
| | `%g` match group                                              | ❌         |                                                                   | | ||||
| | Removal of `%z` match group                                   | ❌         |                                                                   | | ||||
| | Removed `table.maxn`                                          | 🔶         | Only if `disable_lua51_features` is enabled in the configuration. | | ||||
| | `table.pack`/`table.unpack`                                   | ✔          |                                                                   | | ||||
| | `math.log` base argument                                      | ✔          |                                                                   | | ||||
| | Removed `math.log10`                                          | 🔶         | Only if `disable_lua51_features` is enabled in the configuration. | | ||||
| | `*L` mode to `file:read`                                      | ✔          |                                                                   | | ||||
| | `os.execute` exit type + return value                         | ❌         | `os.execute` does not exist in CC:T.                              | | ||||
| | `os.exit` close argument                                      | ❌         | `os.exit` does not exist in CC:T.                                 | | ||||
| | `istailcall` field in `debug.getinfo`                         | ❌         |                                                                   | | ||||
| | `nparams` field in `debug.getinfo`                            | ✔          |                                                                   | | ||||
| | `isvararg` field in `debug.getinfo`                           | ✔          |                                                                   | | ||||
| | `debug.getlocal` negative indices for varargs                 | ❌         |                                                                   | | ||||
| | `debug.getuservalue`/`debug.setuservalue`                     | ❌         | Userdata are rarely used in CC:T, so this is not necessary.       | | ||||
| | `debug.upvalueid`                                             | ✔          |                                                                   | | ||||
| | `debug.upvaluejoin`                                           | ✔          |                                                                   | | ||||
| | Tail call hooks                                               | ❌         |                                                                   | | ||||
| | `=` prefix for chunks                                         | ✔          |                                                                   | | ||||
| | Yield across C boundary                                       | ✔          |                                                                   | | ||||
| | Removal of ambiguity error                                    | ❌         |                                                                   | | ||||
| | Identifiers may no longer use locale-dependent letters        | ✔          |                                                                   | | ||||
| | Ephemeron tables                                              | ❌         |                                                                   | | ||||
| | Identical functions may be reused                             | ❌         |                                                                   | | ||||
| | Generational garbage collector                                | ❌         | Cobalt uses the built-in Java garbage collector.                  | | ||||
|  | ||||
| ## Lua 5.3 | ||||
| | Feature                                                                               | Supported? | Notes                     | | ||||
| |---------------------------------------------------------------------------------------|------------|---------------------------| | ||||
| | Integer subtype                                                                       | ❌         |                           | | ||||
| | Bitwise operators/floor division                                                      | ❌         |                           | | ||||
| | `\u{XXX}` escape sequence                                                             | ✔          |                           | | ||||
| | `utf8` library                                                                        | ✔          |                           | | ||||
| | removed `__ipairs` metamethod                                                         | ✔          |                           | | ||||
| | `coroutine.isyieldable`                                                               | ❌         |                           | | ||||
| | `string.dump` strip argument                                                          | ✔          |                           | | ||||
| | `string.pack`/`string.unpack`/`string.packsize`                                       | ✔          |                           | | ||||
| | `table.move`                                                                          | ❌         |                           | | ||||
| | `math.atan2` -> `math.atan`                                                           | ❌         |                           | | ||||
| | Removed `math.frexp`, `math.ldexp`, `math.pow`, `math.cosh`, `math.sinh`, `math.tanh` | ❌         |                           | | ||||
| | `math.maxinteger`/`math.mininteger`                                                   | ❌         |                           | | ||||
| | `math.tointeger`                                                                      | ❌         |                           | | ||||
| | `math.type`                                                                           | ❌         |                           | | ||||
| | `math.ult`                                                                            | ❌         |                           | | ||||
| | Removed `bit32` library                                                               | ❌         |                           | | ||||
| | Remove `*` from `file:read` modes                                                     | ✔          |                           | | ||||
| | Metamethods respected in `table.*`, `ipairs`                                          | 🔶         | Only `__lt` is respected. | | ||||
|  | ||||
| ## Lua 5.0 | ||||
| | Feature                          | Supported? | Notes                                            | | ||||
| |----------------------------------|------------|--------------------------------------------------| | ||||
| | `arg` table                      | 🔶         | Only set in the shell - not used in functions.   | | ||||
| | `string.gfind`                   | ✔          | Equal to `string.gmatch`.                        | | ||||
| | `table.getn`                     | ✔          | Equal to `#tbl`.                                 | | ||||
| | `table.setn`                     | ❌         |                                                  | | ||||
| | `math.mod`                       | ✔          | Equal to `math.fmod`.                            | | ||||
| | `table.foreach`/`table.foreachi` | ✔          |                                                  | | ||||
| | `gcinfo`                         | ❌         | Cobalt uses the built-in Java garbage collector. | | ||||
| @@ -1,34 +0,0 @@ | ||||
| --- @module fs | ||||
|  | ||||
| --- Returns true if a path is mounted to the parent filesystem. | ||||
| -- | ||||
| -- The root filesystem "/" is considered a mount, along with disk folders and | ||||
| -- the rom folder. Other programs (such as network shares) can exstend this to | ||||
| -- make other mount types by correctly assigning their return value for getDrive. | ||||
| -- | ||||
| -- @tparam string path The path to check. | ||||
| -- @treturn boolean If the path is mounted, rather than a normal file/folder. | ||||
| -- @throws If the path does not exist. | ||||
| -- @see getDrive | ||||
| -- @since 1.87.0 | ||||
| function isDriveRoot(path) end | ||||
|  | ||||
| --[[- Provides completion for a file or directory name, suitable for use with | ||||
| @{_G.read}. | ||||
|  | ||||
| When a directory is a possible candidate for completion, two entries are | ||||
| included - one with a trailing slash (indicating that entries within this | ||||
| directory exist) and one without it (meaning this entry is an immediate | ||||
| completion candidate). `include_dirs` can be set to @{false} to only include | ||||
| those with a trailing slash. | ||||
|  | ||||
| @tparam string path The path to complete. | ||||
| @tparam string location The location where paths are resolved from. | ||||
| @tparam[opt] boolean include_files When @{false}, only directories will be | ||||
| included in the returned list. | ||||
| @tparam[opt] boolean include_dirs When @{false}, "raw" directories will not be | ||||
| included in the returned list. | ||||
| @treturn { string... } A list of possible completion candidates. | ||||
| @since 1.74 | ||||
| ]] | ||||
| function complete(path, location, include_files, include_dirs) end | ||||
| @@ -1,138 +0,0 @@ | ||||
| --[[- | ||||
| Functions in the global environment, defined in `bios.lua`. This does not | ||||
| include standard Lua functions. | ||||
|  | ||||
| @module _G | ||||
| ]] | ||||
|  | ||||
| --[[- Pauses execution for the specified number of seconds. | ||||
|  | ||||
| As it waits for a fixed amount of world ticks, `time` will automatically be | ||||
| rounded up to the nearest multiple of 0.05 seconds. If you are using coroutines | ||||
| or the @{parallel|parallel API}, it will only pause execution of the current | ||||
| thread, not the whole program. | ||||
|  | ||||
| :::tip | ||||
| Because sleep internally uses timers, it is a function that yields. This means | ||||
| that you can use it to prevent "Too long without yielding" errors, however, as | ||||
| the minimum sleep time is 0.05 seconds, it will slow your program down. | ||||
| ::: | ||||
|  | ||||
| :::caution | ||||
| Internally, this function queues and waits for a timer event (using | ||||
| @{os.startTimer}), however it does not listen for any other events. This means | ||||
| that any event that occurs while sleeping will be entirely discarded. If you | ||||
| need to receive events while sleeping, consider using @{os.startTimer|timers}, | ||||
| or the @{parallel|parallel API}. | ||||
| ::: | ||||
|  | ||||
| @tparam number time The number of seconds to sleep for, rounded up to the | ||||
| nearest multiple of 0.05. | ||||
|  | ||||
| @see os.startTimer | ||||
| @usage Sleep for three seconds. | ||||
|  | ||||
|     print("Sleeping for three seconds") | ||||
|     sleep(3) | ||||
|     print("Done!") | ||||
| ]] | ||||
| function sleep(time) end | ||||
|  | ||||
| --- Writes a line of text to the screen without a newline at the end, wrapping | ||||
| -- text if necessary. | ||||
| -- | ||||
| -- @tparam string text The text to write to the string | ||||
| -- @treturn number The number of lines written | ||||
| -- @see print A wrapper around write that adds a newline and accepts multiple arguments | ||||
| -- @usage write("Hello, world") | ||||
| function write(text) end | ||||
|  | ||||
| --- Prints the specified values to the screen separated by spaces, wrapping if | ||||
| -- necessary. After printing, the cursor is moved to the next line. | ||||
| -- | ||||
| -- @param ... The values to print on the screen | ||||
| -- @treturn number The number of lines written | ||||
| -- @usage print("Hello, world!") | ||||
| function print(...) end | ||||
|  | ||||
| --- Prints the specified values to the screen in red, separated by spaces, | ||||
| -- wrapping if necessary. After printing, the cursor is moved to the next line. | ||||
| -- | ||||
| -- @param ... The values to print on the screen | ||||
| -- @usage printError("Something went wrong!") | ||||
| function printError(...) end | ||||
|  | ||||
| --[[- Reads user input from the terminal. This automatically handles arrow keys, | ||||
| pasting, character replacement, history scrollback, auto-completion, and | ||||
| default values. | ||||
|  | ||||
| @tparam[opt] string replaceChar A character to replace each typed character with. | ||||
| This can be used for hiding passwords, for example. | ||||
| @tparam[opt] table history A table holding history items that can be scrolled | ||||
| back to with the up/down arrow keys. The oldest item is at index 1, while the | ||||
| newest item is at the highest index. | ||||
| @tparam[opt] function(partial: string):({ string... }|nil) completeFn A function | ||||
| to be used for completion. This function should take the partial text typed so | ||||
| far, and returns a list of possible completion options. | ||||
| @tparam[opt] string default Default text which should already be entered into | ||||
| the prompt. | ||||
|  | ||||
| @treturn string The text typed in. | ||||
|  | ||||
| @see cc.completion For functions to help with completion. | ||||
| @usage Read a string and echo it back to the user | ||||
|  | ||||
|     write("> ") | ||||
|     local msg = read() | ||||
|     print(msg) | ||||
|  | ||||
| @usage Prompt a user for a password. | ||||
|  | ||||
|     while true do | ||||
|       write("Password> ") | ||||
|       local pwd = read("*") | ||||
|       if pwd == "let me in" then break end | ||||
|       print("Incorrect password, try again.") | ||||
|     end | ||||
|     print("Logged in!") | ||||
|  | ||||
| @usage A complete example with completion, history and a default value. | ||||
|  | ||||
|     local completion = require "cc.completion" | ||||
|     local history = { "potato", "orange", "apple" } | ||||
|     local choices = { "apple", "orange", "banana", "strawberry" } | ||||
|     write("> ") | ||||
|     local msg = read(nil, history, function(text) return completion.choice(text, choices) end, "app") | ||||
|     print(msg) | ||||
|  | ||||
| @changed 1.74 Added `completeFn` parameter. | ||||
| @changed 1.80pr1 Added `default` parameter. | ||||
| ]] | ||||
| function read(replaceChar, history, completeFn, default) end | ||||
|  | ||||
| --- Stores the current ComputerCraft and Minecraft versions. | ||||
| -- | ||||
| -- Outside of Minecraft (for instance, in an emulator) @{_HOST} will contain the | ||||
| -- emulator's version instead. | ||||
| -- | ||||
| -- For example, `ComputerCraft 1.93.0 (Minecraft 1.15.2)`. | ||||
| -- @usage Print the current computer's environment. | ||||
| -- | ||||
| --     print(_HOST) | ||||
| -- @since 1.76 | ||||
| _HOST = _HOST | ||||
|  | ||||
| --[[- The default computer settings as defined in the ComputerCraft | ||||
| configuration. | ||||
|  | ||||
| This is a comma-separated list of settings pairs defined by the mod | ||||
| configuration or server owner. By default, it is empty. | ||||
|  | ||||
| An example value to disable autocompletion: | ||||
|  | ||||
|     shell.autocomplete=false,lua.autocomplete=false,edit.autocomplete=false | ||||
|  | ||||
| @usage _CC_DEFAULT_SETTINGS | ||||
| @since 1.77 | ||||
| ]] | ||||
| _CC_DEFAULT_SETTINGS = _CC_DEFAULT_SETTINGS | ||||
| @@ -1,177 +0,0 @@ | ||||
| --- Make HTTP requests, sending and receiving data to a remote web server. | ||||
| -- | ||||
| -- @module http | ||||
| -- @since 1.1 | ||||
| -- @see local_ips To allow accessing servers running on your local network. | ||||
|  | ||||
| --- Asynchronously make a HTTP request to the given url. | ||||
| -- | ||||
| -- This returns immediately, a @{http_success} or @{http_failure} will be queued | ||||
| -- once the request has completed. | ||||
| -- | ||||
| -- @tparam      string url   The url to request | ||||
| -- @tparam[opt] string body  An optional string containing the body of the | ||||
| -- request. If specified, a `POST` request will be made instead. | ||||
| -- @tparam[opt] { [string] = string } headers Additional headers to send as part | ||||
| -- of this request. | ||||
| -- @tparam[opt] boolean binary Whether to make a binary HTTP request. If true, | ||||
| -- the body will not be UTF-8 encoded, and the received response will not be | ||||
| -- decoded. | ||||
| -- | ||||
| -- @tparam[2] { | ||||
| --   url = string, body? = string, headers? = { [string] = string }, | ||||
| --   binary? = boolean, method? = string, redirect? = boolean, | ||||
| -- } request Options for the request. | ||||
| -- | ||||
| -- This table form is an expanded version of the previous syntax. All arguments | ||||
| -- from above are passed in as fields instead (for instance, | ||||
| -- `http.request("https://example.com")` becomes `http.request { url = | ||||
| -- "https://example.com" }`). | ||||
| -- | ||||
| -- This table also accepts several additional options: | ||||
| -- | ||||
| --  - `method`: Which HTTP method to use, for instance `"PATCH"` or `"DELETE"`. | ||||
| --  - `redirect`: Whether to follow HTTP redirects. Defaults to true. | ||||
| -- | ||||
| -- @see http.get  For a synchronous way to make GET requests. | ||||
| -- @see http.post For a synchronous way to make POST requests. | ||||
| -- | ||||
| -- @changed 1.63 Added argument for headers. | ||||
| -- @changed 1.80pr1 Added argument for binary handles. | ||||
| -- @changed 1.80pr1.6 Added support for table argument. | ||||
| -- @changed 1.86.0 Added PATCH and TRACE methods. | ||||
| function request(...) end | ||||
|  | ||||
| --- Make a HTTP GET request to the given url. | ||||
| -- | ||||
| -- @tparam string url   The url to request | ||||
| -- @tparam[opt] { [string] = string } headers Additional headers to send as part | ||||
| -- of this request. | ||||
| -- @tparam[opt] boolean binary Whether to make a binary HTTP request. If true, | ||||
| -- the body will not be UTF-8 encoded, and the received response will not be | ||||
| -- decoded. | ||||
| -- | ||||
| -- @tparam[2] { | ||||
| --   url = string, headers? = { [string] = string }, | ||||
| --   binary? = boolean, method? = string, redirect? = boolean, | ||||
| -- } request Options for the request. See @{http.request} for details on how | ||||
| -- these options behave. | ||||
| -- | ||||
| -- @treturn Response The resulting http response, which can be read from. | ||||
| -- @treturn[2] nil When the http request failed, such as in the event of a 404 | ||||
| -- error or connection timeout. | ||||
| -- @treturn string A message detailing why the request failed. | ||||
| -- @treturn Response|nil The failing http response, if available. | ||||
| -- | ||||
| -- @changed 1.63 Added argument for headers. | ||||
| -- @changed 1.80pr1 Response handles are now returned on error if available. | ||||
| -- @changed 1.80pr1 Added argument for binary handles. | ||||
| -- @changed 1.80pr1.6 Added support for table argument. | ||||
| -- @changed 1.86.0 Added PATCH and TRACE methods. | ||||
| -- | ||||
| -- @usage Make a request to [example.tweaked.cc](https://example.tweaked.cc), | ||||
| -- and print the returned page. | ||||
| -- ```lua | ||||
| -- local request = http.get("https://example.tweaked.cc") | ||||
| -- print(request.readAll()) | ||||
| -- -- => HTTP is working! | ||||
| -- request.close() | ||||
| -- ``` | ||||
| function get(...) end | ||||
|  | ||||
| --- Make a HTTP POST request to the given url. | ||||
| -- | ||||
| -- @tparam string url   The url to request | ||||
| -- @tparam string body  The body of the POST request. | ||||
| -- @tparam[opt] { [string] = string } headers Additional headers to send as part | ||||
| -- of this request. | ||||
| -- @tparam[opt] boolean binary Whether to make a binary HTTP request. If true, | ||||
| -- the body will not be UTF-8 encoded, and the received response will not be | ||||
| -- decoded. | ||||
| -- | ||||
| -- @tparam[2] { | ||||
| --   url = string, body? = string, headers? = { [string] = string }, | ||||
| --   binary? = boolean, method? = string, redirect? = boolean, | ||||
| -- } request Options for the request. See @{http.request} for details on how | ||||
| -- these options behave. | ||||
| -- | ||||
| -- @treturn Response The resulting http response, which can be read from. | ||||
| -- @treturn[2] nil When the http request failed, such as in the event of a 404 | ||||
| -- error or connection timeout. | ||||
| -- @treturn string A message detailing why the request failed. | ||||
| -- @treturn Response|nil The failing http response, if available. | ||||
| -- | ||||
| -- @since 1.31 | ||||
| -- @changed 1.63 Added argument for headers. | ||||
| -- @changed 1.80pr1 Response handles are now returned on error if available. | ||||
| -- @changed 1.80pr1 Added argument for binary handles. | ||||
| -- @changed 1.80pr1.6 Added support for table argument. | ||||
| -- @changed 1.86.0 Added PATCH and TRACE methods. | ||||
| function post(...) end | ||||
|  | ||||
| --- Asynchronously determine whether a URL can be requested. | ||||
| -- | ||||
| -- If this returns `true`, one should also listen for @{http_check} which will | ||||
| -- container further information about whether the URL is allowed or not. | ||||
| -- | ||||
| -- @tparam string url The URL to check. | ||||
| -- @treturn true When this url is not invalid. This does not imply that it is | ||||
| -- allowed - see the comment above. | ||||
| -- @treturn[2] false When this url is invalid. | ||||
| -- @treturn string A reason why this URL is not valid (for instance, if it is | ||||
| -- malformed, or blocked). | ||||
| -- | ||||
| -- @see http.checkURL For a synchronous version. | ||||
| function checkURLAsync(url) end | ||||
|  | ||||
| --- Determine whether a URL can be requested. | ||||
| -- | ||||
| -- If this returns `true`, one should also listen for @{http_check} which will | ||||
| -- container further information about whether the URL is allowed or not. | ||||
| -- | ||||
| -- @tparam string url The URL to check. | ||||
| -- @treturn true When this url is valid and can be requested via @{http.request}. | ||||
| -- @treturn[2] false When this url is invalid. | ||||
| -- @treturn string A reason why this URL is not valid (for instance, if it is | ||||
| -- malformed, or blocked). | ||||
| -- | ||||
| -- @see http.checkURLAsync For an asynchronous version. | ||||
| -- | ||||
| -- @usage | ||||
| -- ```lua | ||||
| -- print(http.checkURL("https://example.tweaked.cc/")) | ||||
| -- -- => true | ||||
| -- print(http.checkURL("http://localhost/")) | ||||
| -- -- => false Domain not permitted | ||||
| -- print(http.checkURL("not a url")) | ||||
| -- -- => false URL malformed | ||||
| -- ``` | ||||
| function checkURL(url) end | ||||
|  | ||||
| --- Open a websocket. | ||||
| -- | ||||
| -- @tparam string url The websocket url to connect to. This should have the | ||||
| -- `ws://` or `wss://` protocol. | ||||
| -- @tparam[opt] { [string] = string } headers Additional headers to send as part | ||||
| -- of the initial websocket connection. | ||||
| -- | ||||
| -- @treturn Websocket The websocket connection. | ||||
| -- @treturn[2] false If the websocket connection failed. | ||||
| -- @treturn string An error message describing why the connection failed. | ||||
| -- @since 1.80pr1.1 | ||||
| -- @changed 1.80pr1.3 No longer asynchronous. | ||||
| -- @changed 1.95.3 Added User-Agent to default headers. | ||||
| function websocket(url, headers) end | ||||
|  | ||||
| --- Asynchronously open a websocket. | ||||
| -- | ||||
| -- This returns immediately, a @{websocket_success} or @{websocket_failure} | ||||
| -- will be queued once the request has completed. | ||||
| -- | ||||
| -- @tparam string url The websocket url to connect to. This should have the | ||||
| -- `ws://` or `wss://` protocol. | ||||
| -- @tparam[opt] { [string] = string } headers Additional headers to send as part | ||||
| -- of the initial websocket connection. | ||||
| -- @since 1.80pr1.3 | ||||
| -- @changed 1.95.3 Added User-Agent to default headers. | ||||
| function websocketAsync(url, headers) end | ||||
							
								
								
									
										128
									
								
								doc/stub/os.lua
									
									
									
									
									
								
							
							
						
						| @@ -1,128 +0,0 @@ | ||||
| -- Defined in bios.lua | ||||
|  | ||||
| --[[- Loads the given API into the global environment. | ||||
|  | ||||
| This function loads and executes the file at the given path, and all global | ||||
| variables and functions exported by it will by available through the use of | ||||
| `myAPI.<function name>`, where `myAPI` is the base name of the API file. | ||||
|  | ||||
| @tparam string path The path of the API to load. | ||||
| @treturn boolean Whether or not the API was successfully loaded. | ||||
| @since 1.2 | ||||
|  | ||||
| @deprecated When possible it's best to avoid using this function. It pollutes | ||||
| the global table and can mask errors. | ||||
|  | ||||
| @{require} should be used to load libraries instead. | ||||
| ]] | ||||
| function loadAPI(path) end | ||||
|  | ||||
| --- Unloads an API which was loaded by @{os.loadAPI}. | ||||
| -- | ||||
| -- This effectively removes the specified table from `_G`. | ||||
| -- | ||||
| -- @tparam string name The name of the API to unload. | ||||
| -- @since 1.2 | ||||
| -- @deprecated See @{os.loadAPI} for why. | ||||
| function unloadAPI(name) end | ||||
|  | ||||
| --[[- Pause execution of the current thread and waits for any events matching | ||||
| `filter`. | ||||
|  | ||||
| This function @{coroutine.yield|yields} the current process and waits for it | ||||
| to be resumed with a vararg list where the first element matches `filter`. | ||||
| If no `filter` is supplied, this will match all events. | ||||
|  | ||||
| Unlike @{os.pullEventRaw}, it will stop the application upon a "terminate" | ||||
| event, printing the error "Terminated". | ||||
|  | ||||
| @tparam[opt] string filter Event to filter for. | ||||
| @treturn string event The name of the event that fired. | ||||
| @treturn any param... Optional additional parameters of the event. | ||||
| @usage Listen for `mouse_click` events. | ||||
|  | ||||
|     while true do | ||||
|         local event, button, x, y = os.pullEvent("mouse_click") | ||||
|         print("Button", button, "was clicked at", x, ",", y) | ||||
|     end | ||||
|  | ||||
| @usage Listen for multiple events. | ||||
|  | ||||
|     while true do | ||||
|         local eventData = {os.pullEvent()} | ||||
|         local event = eventData[1] | ||||
|  | ||||
|         if event == "mouse_click" then | ||||
|             print("Button", eventData[2], "was clicked at", eventData[3], ",", eventData[4]) | ||||
|         elseif event == "key" then | ||||
|             print("Key code", eventData[2], "was pressed") | ||||
|         end | ||||
|     end | ||||
|  | ||||
| @see os.pullEventRaw To pull the terminate event. | ||||
| @changed 1.3 Added filter argument. | ||||
| ]] | ||||
| function pullEvent(filter) end | ||||
|  | ||||
| --[[- Pause execution of the current thread and waits for events, including the | ||||
| `terminate` event. | ||||
|  | ||||
| This behaves almost the same as @{os.pullEvent}, except it allows you to handle | ||||
| the `terminate` event yourself - the program will not stop execution when | ||||
| <kbd>Ctrl+T</kbd> is pressed. | ||||
|  | ||||
| @tparam[opt] string filter Event to filter for. | ||||
| @treturn string event The name of the event that fired. | ||||
| @treturn any param... Optional additional parameters of the event. | ||||
| @usage Listen for `terminate` events. | ||||
|  | ||||
|     while true do | ||||
|         local event = os.pullEventRaw() | ||||
|         if event == "terminate" then | ||||
|             print("Caught terminate event!") | ||||
|         end | ||||
|     end | ||||
|  | ||||
| @see os.pullEvent To pull events normally. | ||||
| ]] | ||||
| function pullEventRaw(filter) end | ||||
|  | ||||
| --- Pauses execution for the specified number of seconds, alias of @{_G.sleep}. | ||||
| -- | ||||
| -- @tparam number time The number of seconds to sleep for, rounded up to the | ||||
| -- nearest multiple of 0.05. | ||||
| function sleep(time) end | ||||
|  | ||||
| --- Get the current CraftOS version (for example, `CraftOS 1.8`). | ||||
| -- | ||||
| -- This is defined by `bios.lua`. For the current version of CC:Tweaked, this | ||||
| -- should return `CraftOS 1.8`. | ||||
| -- | ||||
| -- @treturn string The current CraftOS version. | ||||
| -- @usage os.version() | ||||
| function version() end | ||||
|  | ||||
| --[[- Run the program at the given path with the specified environment and | ||||
| arguments. | ||||
|  | ||||
| This function does not resolve program names like the shell does. This means | ||||
| that, for example, `os.run("edit")` will not work. As well as this, it does not | ||||
| provide access to the @{shell} API in the environment. For this behaviour, use | ||||
| @{shell.run} instead. | ||||
|  | ||||
| If the program cannot be found, or failed to run, it will print the error and | ||||
| return `false`. If you want to handle this more gracefully, use an alternative | ||||
| such as @{loadfile}. | ||||
|  | ||||
| @tparam table env The environment to run the program with. | ||||
| @tparam string path The exact path of the program to run. | ||||
| @param ... The arguments to pass to the program. | ||||
| @treturn boolean Whether or not the program ran successfully. | ||||
| @usage Run the default shell from within your program: | ||||
|  | ||||
|     os.run({}, "/rom/programs/shell.lua") | ||||
|  | ||||
| @see shell.run | ||||
| @see loadfile | ||||
| ]] | ||||
| function run(env, path, ...) end | ||||
| @@ -1,14 +0,0 @@ | ||||
| --[[- Craft a recipe based on the turtle's inventory. | ||||
|  | ||||
| The turtle's inventory should set up like a crafting grid. For instance, to | ||||
| craft sticks, slots 1 and 5 should contain planks. _All_ other slots should be | ||||
| empty, including those outside the crafting "grid". | ||||
|  | ||||
| @tparam[opt=64] number limit The maximum number of crafting steps to run. | ||||
| @throws When limit is less than 1 or greater than 64. | ||||
| @treturn[1] true If crafting succeeds. | ||||
| @treturn[2] false If crafting fails. | ||||
| @treturn string A string describing why crafting failed. | ||||
| @since 1.4 | ||||
| ]] | ||||
| function craft(limit) end | ||||
| @@ -1,11 +1,17 @@ | ||||
| org.gradle.jvmargs=-Xmx3G | ||||
| kotlin.stdlib.default.dependency=false | ||||
| # Done to increase the memory available to gradle. | ||||
| org.gradle.jvmargs=-Xmx1G | ||||
|  | ||||
| # Mod properties | ||||
| mod_version=1.100.8 | ||||
| mod_version=1.92.0 | ||||
|  | ||||
| # Minecraft properties (update mods.toml when changing) | ||||
| mc_version=1.16.5 | ||||
| mapping_version=2021.08.08 | ||||
| forge_version=36.2.34 | ||||
| # NO SERIOUSLY, UPDATE mods.toml WHEN CHANGING | ||||
| # Minecraft properties | ||||
| mc_version=1.16.3 | ||||
| mappings_version=31 | ||||
|  | ||||
| # Dependencies | ||||
| cloth_config_version=4.8.1 | ||||
| fabric_api_version=0.19.0+build.398-1.16 | ||||
| fabric_loader_version=0.9.2+build.206 | ||||
| jankson_version=1.2.0 | ||||
| modmenu_version=1.14.6+ | ||||
| cloth_api_version=1.4.5 | ||||
|   | ||||
							
								
								
									
										
											BIN
										
									
								
								gradle/wrapper/gradle-wrapper.jar
									
									
									
									
										vendored
									
									
								
							
							
						
						
							
								
								
									
										5
									
								
								gradle/wrapper/gradle-wrapper.properties
									
									
									
									
										vendored
									
									
								
							
							
						
						| @@ -1,5 +1,6 @@ | ||||
| #Tue Jul 07 13:15:43 EDT 2020 | ||||
| distributionUrl=https\://services.gradle.org/distributions/gradle-6.6.1-all.zip | ||||
| distributionBase=GRADLE_USER_HOME | ||||
| distributionPath=wrapper/dists | ||||
| distributionUrl=https\://services.gradle.org/distributions/gradle-7.4.2-bin.zip | ||||
| zipStoreBase=GRADLE_USER_HOME | ||||
| zipStorePath=wrapper/dists | ||||
| zipStoreBase=GRADLE_USER_HOME | ||||
|   | ||||
							
								
								
									
										286
									
								
								gradlew
									
									
									
									
										vendored
									
									
								
							
							
						
						| @@ -1,129 +1,78 @@ | ||||
| #!/bin/sh | ||||
|  | ||||
| # | ||||
| # Copyright © 2015-2021 the original authors. | ||||
| # | ||||
| # Licensed under the Apache License, Version 2.0 (the "License"); | ||||
| # you may not use this file except in compliance with the License. | ||||
| # You may obtain a copy of the License at | ||||
| # | ||||
| #      https://www.apache.org/licenses/LICENSE-2.0 | ||||
| # | ||||
| # Unless required by applicable law or agreed to in writing, software | ||||
| # distributed under the License is distributed on an "AS IS" BASIS, | ||||
| # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. | ||||
| # See the License for the specific language governing permissions and | ||||
| # limitations under the License. | ||||
| # | ||||
| #!/usr/bin/env sh | ||||
|  | ||||
| ############################################################################## | ||||
| # | ||||
| #   Gradle start up script for POSIX generated by Gradle. | ||||
| # | ||||
| #   Important for running: | ||||
| # | ||||
| #   (1) You need a POSIX-compliant shell to run this script. If your /bin/sh is | ||||
| #       noncompliant, but you have some other compliant shell such as ksh or | ||||
| #       bash, then to run this script, type that shell name before the whole | ||||
| #       command line, like: | ||||
| # | ||||
| #           ksh Gradle | ||||
| # | ||||
| #       Busybox and similar reduced shells will NOT work, because this script | ||||
| #       requires all of these POSIX shell features: | ||||
| #         * functions; | ||||
| #         * expansions «$var», «${var}», «${var:-default}», «${var+SET}», | ||||
| #           «${var#prefix}», «${var%suffix}», and «$( cmd )»; | ||||
| #         * compound commands having a testable exit status, especially «case»; | ||||
| #         * various built-in commands including «command», «set», and «ulimit». | ||||
| # | ||||
| #   Important for patching: | ||||
| # | ||||
| #   (2) This script targets any POSIX shell, so it avoids extensions provided | ||||
| #       by Bash, Ksh, etc; in particular arrays are avoided. | ||||
| # | ||||
| #       The "traditional" practice of packing multiple parameters into a | ||||
| #       space-separated string is a well documented source of bugs and security | ||||
| #       problems, so this is (mostly) avoided, by progressively accumulating | ||||
| #       options in "$@", and eventually passing that to Java. | ||||
| # | ||||
| #       Where the inherited environment variables (DEFAULT_JVM_OPTS, JAVA_OPTS, | ||||
| #       and GRADLE_OPTS) rely on word-splitting, this is performed explicitly; | ||||
| #       see the in-line comments for details. | ||||
| # | ||||
| #       There are tweaks for specific operating systems such as AIX, CygWin, | ||||
| #       Darwin, MinGW, and NonStop. | ||||
| # | ||||
| #   (3) This script is generated from the Groovy template | ||||
| #       https://github.com/gradle/gradle/blob/master/subprojects/plugins/src/main/resources/org/gradle/api/internal/plugins/unixStartScript.txt | ||||
| #       within the Gradle project. | ||||
| # | ||||
| #       You can find Gradle at https://github.com/gradle/gradle/. | ||||
| # | ||||
| ## | ||||
| ##  Gradle start up script for UN*X | ||||
| ## | ||||
| ############################################################################## | ||||
|  | ||||
| # Attempt to set APP_HOME | ||||
|  | ||||
| # Resolve links: $0 may be a link | ||||
| app_path=$0 | ||||
|  | ||||
| # Need this for daisy-chained symlinks. | ||||
| while | ||||
|     APP_HOME=${app_path%"${app_path##*/}"}  # leaves a trailing /; empty if no leading path | ||||
|     [ -h "$app_path" ] | ||||
| do | ||||
|     ls=$( ls -ld "$app_path" ) | ||||
|     link=${ls#*' -> '} | ||||
|     case $link in             #( | ||||
|       /*)   app_path=$link ;; #( | ||||
|       *)    app_path=$APP_HOME$link ;; | ||||
|     esac | ||||
| PRG="$0" | ||||
| # Need this for relative symlinks. | ||||
| while [ -h "$PRG" ] ; do | ||||
|     ls=`ls -ld "$PRG"` | ||||
|     link=`expr "$ls" : '.*-> \(.*\)$'` | ||||
|     if expr "$link" : '/.*' > /dev/null; then | ||||
|         PRG="$link" | ||||
|     else | ||||
|         PRG=`dirname "$PRG"`"/$link" | ||||
|     fi | ||||
| done | ||||
|  | ||||
| APP_HOME=$( cd "${APP_HOME:-./}" && pwd -P ) || exit | ||||
| SAVED="`pwd`" | ||||
| cd "`dirname \"$PRG\"`/" >/dev/null | ||||
| APP_HOME="`pwd -P`" | ||||
| cd "$SAVED" >/dev/null | ||||
|  | ||||
| APP_NAME="Gradle" | ||||
| APP_BASE_NAME=${0##*/} | ||||
| APP_BASE_NAME=`basename "$0"` | ||||
|  | ||||
| # Add default JVM options here. You can also use JAVA_OPTS and GRADLE_OPTS to pass JVM options to this script. | ||||
| DEFAULT_JVM_OPTS='"-Xmx64m" "-Xms64m"' | ||||
| DEFAULT_JVM_OPTS="" | ||||
|  | ||||
| # Use the maximum available, or set MAX_FD != -1 to use that value. | ||||
| MAX_FD=maximum | ||||
| MAX_FD="maximum" | ||||
|  | ||||
| warn () { | ||||
|     echo "$*" | ||||
| } >&2 | ||||
| } | ||||
|  | ||||
| die () { | ||||
|     echo | ||||
|     echo "$*" | ||||
|     echo | ||||
|     exit 1 | ||||
| } >&2 | ||||
| } | ||||
|  | ||||
| # OS specific support (must be 'true' or 'false'). | ||||
| cygwin=false | ||||
| msys=false | ||||
| darwin=false | ||||
| nonstop=false | ||||
| case "$( uname )" in                #( | ||||
|   CYGWIN* )         cygwin=true  ;; #( | ||||
|   Darwin* )         darwin=true  ;; #( | ||||
|   MSYS* | MINGW* )  msys=true    ;; #( | ||||
|   NONSTOP* )        nonstop=true ;; | ||||
| case "`uname`" in | ||||
|   CYGWIN* ) | ||||
|     cygwin=true | ||||
|     ;; | ||||
|   Darwin* ) | ||||
|     darwin=true | ||||
|     ;; | ||||
|   MINGW* ) | ||||
|     msys=true | ||||
|     ;; | ||||
|   NONSTOP* ) | ||||
|     nonstop=true | ||||
|     ;; | ||||
| esac | ||||
|  | ||||
| CLASSPATH=$APP_HOME/gradle/wrapper/gradle-wrapper.jar | ||||
|  | ||||
|  | ||||
| # Determine the Java command to use to start the JVM. | ||||
| if [ -n "$JAVA_HOME" ] ; then | ||||
|     if [ -x "$JAVA_HOME/jre/sh/java" ] ; then | ||||
|         # IBM's JDK on AIX uses strange locations for the executables | ||||
|         JAVACMD=$JAVA_HOME/jre/sh/java | ||||
|         JAVACMD="$JAVA_HOME/jre/sh/java" | ||||
|     else | ||||
|         JAVACMD=$JAVA_HOME/bin/java | ||||
|         JAVACMD="$JAVA_HOME/bin/java" | ||||
|     fi | ||||
|     if [ ! -x "$JAVACMD" ] ; then | ||||
|         die "ERROR: JAVA_HOME is set to an invalid directory: $JAVA_HOME | ||||
| @@ -132,7 +81,7 @@ Please set the JAVA_HOME variable in your environment to match the | ||||
| location of your Java installation." | ||||
|     fi | ||||
| else | ||||
|     JAVACMD=java | ||||
|     JAVACMD="java" | ||||
|     which java >/dev/null 2>&1 || die "ERROR: JAVA_HOME is not set and no 'java' command could be found in your PATH. | ||||
|  | ||||
| Please set the JAVA_HOME variable in your environment to match the | ||||
| @@ -140,95 +89,84 @@ location of your Java installation." | ||||
| fi | ||||
|  | ||||
| # Increase the maximum file descriptors if we can. | ||||
| if ! "$cygwin" && ! "$darwin" && ! "$nonstop" ; then | ||||
|     case $MAX_FD in #( | ||||
|       max*) | ||||
|         MAX_FD=$( ulimit -H -n ) || | ||||
|             warn "Could not query maximum file descriptor limit" | ||||
|     esac | ||||
|     case $MAX_FD in  #( | ||||
|       '' | soft) :;; #( | ||||
|       *) | ||||
|         ulimit -n "$MAX_FD" || | ||||
|             warn "Could not set maximum file descriptor limit to $MAX_FD" | ||||
|     esac | ||||
| fi | ||||
|  | ||||
| # Collect all arguments for the java command, stacking in reverse order: | ||||
| #   * args from the command line | ||||
| #   * the main class name | ||||
| #   * -classpath | ||||
| #   * -D...appname settings | ||||
| #   * --module-path (only if needed) | ||||
| #   * DEFAULT_JVM_OPTS, JAVA_OPTS, and GRADLE_OPTS environment variables. | ||||
|  | ||||
| # For Cygwin or MSYS, switch paths to Windows format before running java | ||||
| if "$cygwin" || "$msys" ; then | ||||
|     APP_HOME=$( cygpath --path --mixed "$APP_HOME" ) | ||||
|     CLASSPATH=$( cygpath --path --mixed "$CLASSPATH" ) | ||||
|  | ||||
|     JAVACMD=$( cygpath --unix "$JAVACMD" ) | ||||
|  | ||||
|     # Now convert the arguments - kludge to limit ourselves to /bin/sh | ||||
|     for arg do | ||||
|         if | ||||
|             case $arg in                                #( | ||||
|               -*)   false ;;                            # don't mess with options #( | ||||
|               /?*)  t=${arg#/} t=/${t%%/*}              # looks like a POSIX filepath | ||||
|                     [ -e "$t" ] ;;                      #( | ||||
|               *)    false ;; | ||||
|             esac | ||||
|         then | ||||
|             arg=$( cygpath --path --ignore --mixed "$arg" ) | ||||
| if [ "$cygwin" = "false" -a "$darwin" = "false" -a "$nonstop" = "false" ] ; then | ||||
|     MAX_FD_LIMIT=`ulimit -H -n` | ||||
|     if [ $? -eq 0 ] ; then | ||||
|         if [ "$MAX_FD" = "maximum" -o "$MAX_FD" = "max" ] ; then | ||||
|             MAX_FD="$MAX_FD_LIMIT" | ||||
|         fi | ||||
|         # Roll the args list around exactly as many times as the number of | ||||
|         # args, so each arg winds up back in the position where it started, but | ||||
|         # possibly modified. | ||||
|         # | ||||
|         # NB: a `for` loop captures its iteration list before it begins, so | ||||
|         # changing the positional parameters here affects neither the number of | ||||
|         # iterations, nor the values presented in `arg`. | ||||
|         shift                   # remove old arg | ||||
|         set -- "$@" "$arg"      # push replacement arg | ||||
|     done | ||||
|         ulimit -n $MAX_FD | ||||
|         if [ $? -ne 0 ] ; then | ||||
|             warn "Could not set maximum file descriptor limit: $MAX_FD" | ||||
|         fi | ||||
|     else | ||||
|         warn "Could not query maximum file descriptor limit: $MAX_FD_LIMIT" | ||||
|     fi | ||||
| fi | ||||
|  | ||||
| # Collect all arguments for the java command; | ||||
| #   * $DEFAULT_JVM_OPTS, $JAVA_OPTS, and $GRADLE_OPTS can contain fragments of | ||||
| #     shell script including quotes and variable substitutions, so put them in | ||||
| #     double quotes to make sure that they get re-expanded; and | ||||
| #   * put everything else in single quotes, so that it's not re-expanded. | ||||
| # For Darwin, add options to specify how the application appears in the dock | ||||
| if $darwin; then | ||||
|     GRADLE_OPTS="$GRADLE_OPTS \"-Xdock:name=$APP_NAME\" \"-Xdock:icon=$APP_HOME/media/gradle.icns\"" | ||||
| fi | ||||
|  | ||||
| set -- \ | ||||
|         "-Dorg.gradle.appname=$APP_BASE_NAME" \ | ||||
|         -classpath "$CLASSPATH" \ | ||||
|         org.gradle.wrapper.GradleWrapperMain \ | ||||
|         "$@" | ||||
| # For Cygwin, switch paths to Windows format before running java | ||||
| if $cygwin ; then | ||||
|     APP_HOME=`cygpath --path --mixed "$APP_HOME"` | ||||
|     CLASSPATH=`cygpath --path --mixed "$CLASSPATH"` | ||||
|     JAVACMD=`cygpath --unix "$JAVACMD"` | ||||
|  | ||||
| # Use "xargs" to parse quoted args. | ||||
| # | ||||
| # With -n1 it outputs one arg per line, with the quotes and backslashes removed. | ||||
| # | ||||
| # In Bash we could simply go: | ||||
| # | ||||
| #   readarray ARGS < <( xargs -n1 <<<"$var" ) && | ||||
| #   set -- "${ARGS[@]}" "$@" | ||||
| # | ||||
| # but POSIX shell has neither arrays nor command substitution, so instead we | ||||
| # post-process each arg (as a line of input to sed) to backslash-escape any | ||||
| # character that might be a shell metacharacter, then use eval to reverse | ||||
| # that process (while maintaining the separation between arguments), and wrap | ||||
| # the whole thing up as a single "set" statement. | ||||
| # | ||||
| # This will of course break if any of these variables contains a newline or | ||||
| # an unmatched quote. | ||||
| # | ||||
|     # We build the pattern for arguments to be converted via cygpath | ||||
|     ROOTDIRSRAW=`find -L / -maxdepth 1 -mindepth 1 -type d 2>/dev/null` | ||||
|     SEP="" | ||||
|     for dir in $ROOTDIRSRAW ; do | ||||
|         ROOTDIRS="$ROOTDIRS$SEP$dir" | ||||
|         SEP="|" | ||||
|     done | ||||
|     OURCYGPATTERN="(^($ROOTDIRS))" | ||||
|     # Add a user-defined pattern to the cygpath arguments | ||||
|     if [ "$GRADLE_CYGPATTERN" != "" ] ; then | ||||
|         OURCYGPATTERN="$OURCYGPATTERN|($GRADLE_CYGPATTERN)" | ||||
|     fi | ||||
|     # Now convert the arguments - kludge to limit ourselves to /bin/sh | ||||
|     i=0 | ||||
|     for arg in "$@" ; do | ||||
|         CHECK=`echo "$arg"|egrep -c "$OURCYGPATTERN" -` | ||||
|         CHECK2=`echo "$arg"|egrep -c "^-"`                                 ### Determine if an option | ||||
|  | ||||
| eval "set -- $( | ||||
|         printf '%s\n' "$DEFAULT_JVM_OPTS $JAVA_OPTS $GRADLE_OPTS" | | ||||
|         xargs -n1 | | ||||
|         sed ' s~[^-[:alnum:]+,./:=@_]~\\&~g; ' | | ||||
|         tr '\n' ' ' | ||||
|     )" '"$@"' | ||||
|         if [ $CHECK -ne 0 ] && [ $CHECK2 -eq 0 ] ; then                    ### Added a condition | ||||
|             eval `echo args$i`=`cygpath --path --ignore --mixed "$arg"` | ||||
|         else | ||||
|             eval `echo args$i`="\"$arg\"" | ||||
|         fi | ||||
|         i=$((i+1)) | ||||
|     done | ||||
|     case $i in | ||||
|         (0) set -- ;; | ||||
|         (1) set -- "$args0" ;; | ||||
|         (2) set -- "$args0" "$args1" ;; | ||||
|         (3) set -- "$args0" "$args1" "$args2" ;; | ||||
|         (4) set -- "$args0" "$args1" "$args2" "$args3" ;; | ||||
|         (5) set -- "$args0" "$args1" "$args2" "$args3" "$args4" ;; | ||||
|         (6) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" ;; | ||||
|         (7) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" "$args6" ;; | ||||
|         (8) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" "$args6" "$args7" ;; | ||||
|         (9) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" "$args6" "$args7" "$args8" ;; | ||||
|     esac | ||||
| fi | ||||
|  | ||||
| # Escape application args | ||||
| save () { | ||||
|     for i do printf %s\\n "$i" | sed "s/'/'\\\\''/g;1s/^/'/;\$s/\$/' \\\\/" ; done | ||||
|     echo " " | ||||
| } | ||||
| APP_ARGS=$(save "$@") | ||||
|  | ||||
| # Collect all arguments for the java command, following the shell quoting and substitution rules | ||||
| eval set -- $DEFAULT_JVM_OPTS $JAVA_OPTS $GRADLE_OPTS "\"-Dorg.gradle.appname=$APP_BASE_NAME\"" -classpath "\"$CLASSPATH\"" org.gradle.wrapper.GradleWrapperMain "$APP_ARGS" | ||||
|  | ||||
| # by default we should be in the correct project dir, but when run from Finder on Mac, the cwd is wrong | ||||
| if [ "$(uname)" = "Darwin" ] && [ "$HOME" = "$PWD" ]; then | ||||
|   cd "$(dirname "$0")" | ||||
| fi | ||||
|  | ||||
| exec "$JAVACMD" "$@" | ||||
|   | ||||
							
								
								
									
										43
									
								
								gradlew.bat
									
									
									
									
										vendored
									
									
								
							
							
						
						| @@ -1,19 +1,3 @@ | ||||
| @rem | ||||
| @rem Copyright 2015 the original author or authors. | ||||
| @rem | ||||
| @rem Licensed under the Apache License, Version 2.0 (the "License"); | ||||
| @rem you may not use this file except in compliance with the License. | ||||
| @rem You may obtain a copy of the License at | ||||
| @rem | ||||
| @rem      https://www.apache.org/licenses/LICENSE-2.0 | ||||
| @rem | ||||
| @rem Unless required by applicable law or agreed to in writing, software | ||||
| @rem distributed under the License is distributed on an "AS IS" BASIS, | ||||
| @rem WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. | ||||
| @rem See the License for the specific language governing permissions and | ||||
| @rem limitations under the License. | ||||
| @rem | ||||
|  | ||||
| @if "%DEBUG%" == "" @echo off | ||||
| @rem ########################################################################## | ||||
| @rem | ||||
| @@ -29,18 +13,15 @@ if "%DIRNAME%" == "" set DIRNAME=. | ||||
| set APP_BASE_NAME=%~n0 | ||||
| set APP_HOME=%DIRNAME% | ||||
|  | ||||
| @rem Resolve any "." and ".." in APP_HOME to make it shorter. | ||||
| for %%i in ("%APP_HOME%") do set APP_HOME=%%~fi | ||||
|  | ||||
| @rem Add default JVM options here. You can also use JAVA_OPTS and GRADLE_OPTS to pass JVM options to this script. | ||||
| set DEFAULT_JVM_OPTS="-Xmx64m" "-Xms64m" | ||||
| set DEFAULT_JVM_OPTS= | ||||
|  | ||||
| @rem Find java.exe | ||||
| if defined JAVA_HOME goto findJavaFromJavaHome | ||||
|  | ||||
| set JAVA_EXE=java.exe | ||||
| %JAVA_EXE% -version >NUL 2>&1 | ||||
| if "%ERRORLEVEL%" == "0" goto execute | ||||
| if "%ERRORLEVEL%" == "0" goto init | ||||
|  | ||||
| echo. | ||||
| echo ERROR: JAVA_HOME is not set and no 'java' command could be found in your PATH. | ||||
| @@ -54,7 +35,7 @@ goto fail | ||||
| set JAVA_HOME=%JAVA_HOME:"=% | ||||
| set JAVA_EXE=%JAVA_HOME%/bin/java.exe | ||||
|  | ||||
| if exist "%JAVA_EXE%" goto execute | ||||
| if exist "%JAVA_EXE%" goto init | ||||
|  | ||||
| echo. | ||||
| echo ERROR: JAVA_HOME is set to an invalid directory: %JAVA_HOME% | ||||
| @@ -64,14 +45,28 @@ echo location of your Java installation. | ||||
|  | ||||
| goto fail | ||||
|  | ||||
| :init | ||||
| @rem Get command-line arguments, handling Windows variants | ||||
|  | ||||
| if not "%OS%" == "Windows_NT" goto win9xME_args | ||||
|  | ||||
| :win9xME_args | ||||
| @rem Slurp the command line arguments. | ||||
| set CMD_LINE_ARGS= | ||||
| set _SKIP=2 | ||||
|  | ||||
| :win9xME_args_slurp | ||||
| if "x%~1" == "x" goto execute | ||||
|  | ||||
| set CMD_LINE_ARGS=%* | ||||
|  | ||||
| :execute | ||||
| @rem Setup the command line | ||||
|  | ||||
| set CLASSPATH=%APP_HOME%\gradle\wrapper\gradle-wrapper.jar | ||||
|  | ||||
|  | ||||
| @rem Execute Gradle | ||||
| "%JAVA_EXE%" %DEFAULT_JVM_OPTS% %JAVA_OPTS% %GRADLE_OPTS% "-Dorg.gradle.appname=%APP_BASE_NAME%" -classpath "%CLASSPATH%" org.gradle.wrapper.GradleWrapperMain %* | ||||
| "%JAVA_EXE%" %DEFAULT_JVM_OPTS% %JAVA_OPTS% %GRADLE_OPTS% "-Dorg.gradle.appname=%APP_BASE_NAME%" -classpath "%CLASSPATH%" org.gradle.wrapper.GradleWrapperMain %CMD_LINE_ARGS% | ||||
|  | ||||
| :end | ||||
| @rem End local scope for the variables with windows NT shell | ||||
|   | ||||
							
								
								
									
										116
									
								
								illuaminate.sexp
									
									
									
									
									
								
							
							
						
						| @@ -1,116 +0,0 @@ | ||||
| ; -*- mode: Lisp;-*- | ||||
|  | ||||
| (sources | ||||
|   /doc/ | ||||
|   /build/docs/luaJavadoc/ | ||||
|   /src/main/resources/*/computercraft/lua/bios.lua | ||||
|   /src/main/resources/*/computercraft/lua/rom/ | ||||
|   /src/test/resources/test-rom | ||||
|   /src/web/mount) | ||||
|  | ||||
|  | ||||
| (doc | ||||
|   (destination build/docs/lua) | ||||
|   (index doc/index.md) | ||||
|  | ||||
|   (site | ||||
|     (title "CC: Tweaked") | ||||
|     (logo src/main/resources/pack.png) | ||||
|     (url https://tweaked.cc/) | ||||
|     (source-link https://github.com/cc-tweaked/CC-Tweaked/blob/${commit}/${path}#L${line}) | ||||
|  | ||||
|     (styles src/web/styles.css) | ||||
|     (scripts build/rollup/index.js) | ||||
|     (head doc/head.html)) | ||||
|  | ||||
|   (module-kinds | ||||
|     (peripheral Peripherals) | ||||
|     (generic_peripheral "Generic Peripherals") | ||||
|     (event Events) | ||||
|     (guide Guides) | ||||
|     (reference Reference)) | ||||
|  | ||||
|   (library-path | ||||
|     /doc/stub/ | ||||
|     /build/docs/luaJavadoc/ | ||||
|  | ||||
|     /src/main/resources/*/computercraft/lua/rom/apis/ | ||||
|     /src/main/resources/*/computercraft/lua/rom/apis/command/ | ||||
|     /src/main/resources/*/computercraft/lua/rom/apis/turtle/ | ||||
|  | ||||
|     /src/main/resources/*/computercraft/lua/rom/modules/main/ | ||||
|     /src/main/resources/*/computercraft/lua/rom/modules/command/ | ||||
|     /src/main/resources/*/computercraft/lua/rom/modules/turtle/)) | ||||
|  | ||||
| (at / | ||||
|   (linters | ||||
|     syntax:string-index | ||||
|  | ||||
|     ;; It'd be nice to avoid this, but right now there's a lot of instances of | ||||
|     ;; it. | ||||
|     -var:set-loop | ||||
|  | ||||
|     ;; It's useful to name arguments for documentation, so we allow this. It'd | ||||
|     ;; be good to find a compromise in the future, but this works for now. | ||||
|     -var:unused-arg) | ||||
|  | ||||
|   (lint | ||||
|     (bracket-spaces | ||||
|       (call no-space) | ||||
|       (function-args no-space) | ||||
|       (parens no-space) | ||||
|       (table space) | ||||
|       (index no-space)) | ||||
|  | ||||
|     (allow-clarifying-parens true) | ||||
|  | ||||
|     ;; colours imports from colors, and we don't handle that right now. | ||||
|     ;; keys is entirely dynamic, so we skip it. | ||||
|     (dynamic-modules colours keys _G) | ||||
|  | ||||
|     (globals | ||||
|       :max | ||||
|       _CC_DEFAULT_SETTINGS | ||||
|       _CC_DISABLE_LUA51_FEATURES | ||||
|       _HOST | ||||
|       ;; Ideally we'd pick these up from bios.lua, but illuaminate currently | ||||
|       ;; isn't smart enough. | ||||
|       sleep write printError read rs))) | ||||
|  | ||||
| ;; We disable the unused global linter in bios.lua and the APIs. In the future | ||||
| ;; hopefully we'll get illuaminate to handle this. | ||||
| (at | ||||
|   (/src/main/resources/*/computercraft/lua/bios.lua | ||||
|    /src/main/resources/*/computercraft/lua/rom/apis/) | ||||
|   (linters -var:unused-global) | ||||
|   (lint (allow-toplevel-global true))) | ||||
|  | ||||
| ;; Silence some variable warnings in documentation stubs. | ||||
| (at (/doc/stub/ /build/docs/luaJavadoc/) | ||||
|   (linters -var:unused-global) | ||||
|   (lint (allow-toplevel-global true))) | ||||
|  | ||||
| ;; Suppress warnings for currently undocumented modules. | ||||
| (at | ||||
|   (; Lua APIs | ||||
|    /src/main/resources/*/computercraft/lua/rom/apis/io.lua | ||||
|    /src/main/resources/*/computercraft/lua/rom/apis/window.lua) | ||||
|  | ||||
|   (linters -doc:undocumented -doc:undocumented-arg -doc:undocumented-return)) | ||||
|  | ||||
| ;; Suppress warnings for various APIs using its own deprecated members. | ||||
| (at | ||||
|   (/src/main/resources/*/computercraft/lua/bios.lua | ||||
|    /src/main/resources/*/computercraft/lua/rom/apis/turtle/turtle.lua) | ||||
|   (linters -var:deprecated)) | ||||
|  | ||||
| (at /src/test/resources/test-rom | ||||
|   ; We should still be able to test deprecated members. | ||||
|   (linters -var:deprecated) | ||||
|  | ||||
|   (lint | ||||
|     (globals | ||||
|       :max sleep write | ||||
|       cct_test describe expect howlci fail it pending stub))) | ||||
|  | ||||
| (at /src/web/mount/expr_template.lua (lint (globals :max __expr__))) | ||||
							
								
								
									
										2613
									
								
								package-lock.json
									
									
									
										generated
									
									
									
								
							
							
						
						
							
								
								
									
										29
									
								
								package.json
									
									
									
									
									
								
							
							
						
						| @@ -1,29 +0,0 @@ | ||||
| { | ||||
|   "name": "tweaked.cc", | ||||
|   "version": "1.0.0", | ||||
|   "description": "Website additions for tweaked.cc", | ||||
|   "author": "SquidDev", | ||||
|   "license": "BSD-3-Clause", | ||||
|   "type": "module", | ||||
|   "dependencies": { | ||||
|     "preact": "^10.5.5", | ||||
|     "tslib": "^2.0.3" | ||||
|   }, | ||||
|   "devDependencies": { | ||||
|     "@rollup/plugin-typescript": "^8.2.5", | ||||
|     "@rollup/plugin-url": "^6.1.0", | ||||
|     "@types/glob": "^7.2.0", | ||||
|     "@types/react-dom": "^18.0.5", | ||||
|     "glob": "^8.0.3", | ||||
|     "react": "^18.1.0", | ||||
|     "react-dom": "^18.1.0", | ||||
|     "rehype": "^12.0.1", | ||||
|     "rehype-highlight": "^5.0.2", | ||||
|     "rehype-react": "^7.1.1", | ||||
|     "requirejs": "^2.3.6", | ||||
|     "rollup": "^2.33.1", | ||||
|     "rollup-plugin-terser": "^7.0.2", | ||||
|     "ts-node": "^10.8.0", | ||||
|     "typescript": "^4.0.5" | ||||
|   } | ||||
| } | ||||
							
								
								
									
										48
									
								
								patchwork.md
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,48 @@ | ||||
| # Just my list of things I have ported over | ||||
|  | ||||
| Format for the changelog of ported stuff | ||||
| ``` | ||||
| commit    // Shows commit from CC:T | ||||
| commit2   // Shows a commit that is the same thing, just a clean up, only if right after | ||||
| Title     // Commit Title | ||||
| SubScript // Desc of commit | ||||
| ``` | ||||
|  | ||||
| If a edit that is present in CC:T is not needed, I will skip over it. | ||||
| Any and all references to an issue number, are to be found on CC:T's repo. not this oen | ||||
|  | ||||
| ```md | ||||
| 5155e18de279a193c558aa029963486fd1294769 | ||||
| Added translation for Vietnamese | ||||
| Co-authored-by: Boom <boom@flyingpackets.net> | ||||
| ``` | ||||
|  | ||||
| ``` | ||||
| 7e121ff72f2b1504cd6af47b57500876682bac45 | ||||
| ae6124d1f477487abab1858abde8c4ec49dfee3c | ||||
| Translations for Vienamese | ||||
| Co-authored-by: Boom <boom@flyingpackets.net> | ||||
| ``` | ||||
|  | ||||
| ``` | ||||
| 59de21eae29849988e77fad6bc335f5ce78dfec7 | ||||
| Handle tabs when parsing JSON | ||||
| Fixes #539 | ||||
| ``` | ||||
|  | ||||
| ``` | ||||
| 748ebbe66bf0a4239bde34f557e4b4b75d61d990 | ||||
| Bump to 1.92.0 | ||||
| A tiny release, but there's new features so it's technically a minor | ||||
| bump. | ||||
| ``` | ||||
|  | ||||
| Cherry Picked because this update was partially related to forge updates rather than mod updates | ||||
| ``` | ||||
| 8b4a01df27ff7f6fa9ffd9c2188c6e3166edd515 | ||||
| Update to Minecraft 1.16.3 | ||||
|  | ||||
| I hope the Fabric folks now realise this is gonna be a race of who can | ||||
| update first :p. Either way, this was a very easy update - only changes | ||||
| were due to unrelated Forge changes. | ||||
| ``` | ||||
| @@ -1,56 +0,0 @@ | ||||
| import { readFileSync } from "fs"; | ||||
| import path from "path"; | ||||
|  | ||||
| import typescript from "@rollup/plugin-typescript"; | ||||
| import url from '@rollup/plugin-url'; | ||||
| import { terser } from "rollup-plugin-terser"; | ||||
|  | ||||
| const input = "src/web"; | ||||
| const requirejs = readFileSync("node_modules/requirejs/require.js"); | ||||
|  | ||||
| export default { | ||||
|     input: [`${input}/index.tsx`], | ||||
|     output: { | ||||
|         dir: "build/rollup/", | ||||
|         // We bundle requirejs (and config) into the header. It's rather gross | ||||
|         // but also works reasonably well. | ||||
|         // Also suffix a ?v=${date} onto the end in the event we need to require a specific copy-cat version. | ||||
|         banner: ` | ||||
|             ${requirejs} | ||||
|             require.config({ | ||||
|                 paths: { copycat: "https://copy-cat.squiddev.cc" }, | ||||
|                 urlArgs: function(id) { return id == "copycat/embed" ? "?v=20211221" : ""; } | ||||
|             }); | ||||
|         `, | ||||
|         format: "amd", | ||||
|         preferConst: true, | ||||
|         amd: { | ||||
|             define: "require", | ||||
|         } | ||||
|     }, | ||||
|     context: "window", | ||||
|     external: ["copycat/embed"], | ||||
|  | ||||
|     plugins: [ | ||||
|         typescript(), | ||||
|  | ||||
|         url({ | ||||
|             include: "**/*.dfpwm", | ||||
|             fileName: "[name]-[hash][extname]", | ||||
|             publicPath: "/", | ||||
|         }), | ||||
|  | ||||
|         { | ||||
|             name: "cc-tweaked", | ||||
|             async transform(code, file) { | ||||
|                 // Allow loading files in /mount. | ||||
|                 const ext = path.extname(file); | ||||
|                 return ext != '.dfpwm' && path.dirname(file) === path.resolve(`${input}/mount`) | ||||
|                     ? `export default ${JSON.stringify(code)};\n` | ||||
|                     : null; | ||||
|             }, | ||||
|         }, | ||||
|  | ||||
|         terser(), | ||||
|     ], | ||||
| }; | ||||
							
								
								
									
										12
									
								
								settings.gradle
									
									
									
									
									
										Normal file
									
								
							
							
						
						| @@ -0,0 +1,12 @@ | ||||
| pluginManagement { | ||||
|     repositories { | ||||
|         jcenter() | ||||
|         maven { | ||||
|             name = 'Fabric' | ||||
|             url = 'https://maven.fabricmc.net/' | ||||
|         } | ||||
|         gradlePluginPortal() | ||||
|     } | ||||
| } | ||||
|  | ||||
| rootProject.name = "cc-tweaked-fabric-${mc_version}" | ||||
| @@ -1,17 +0,0 @@ | ||||
| pluginManagement { | ||||
|     repositories { | ||||
|         gradlePluginPortal() | ||||
|         maven("https://maven.minecraftforge.net") | ||||
|         maven("https://maven.parchmentmc.org") | ||||
|     } | ||||
|     resolutionStrategy { | ||||
|         eachPlugin { | ||||
|             if (requested.id.id == "org.spongepowered.mixin") { | ||||
|                 useModule("org.spongepowered:mixingradle:${requested.version}") | ||||
|             } | ||||
|         } | ||||
|     } | ||||
| } | ||||
|  | ||||
| val mc_version: String by settings | ||||
| rootProject.name = "cc-tweaked-${mc_version}" | ||||
| @@ -1,760 +0,0 @@ | ||||
| { | ||||
|   "itemNames": { | ||||
|     "computercraft:cable": "Networking Cable", | ||||
|     "computercraft:computer_advanced": "Advanced Computer", | ||||
|     "computercraft:computer_command": "Command Computer", | ||||
|     "computercraft:computer_normal": "Computer", | ||||
|     "computercraft:disk": "Floppy Disk", | ||||
|     "computercraft:disk_drive": "Disk Drive", | ||||
|     "computercraft:monitor_advanced": "Advanced Monitor", | ||||
|     "computercraft:monitor_normal": "Monitor", | ||||
|     "computercraft:pocket_computer_advanced": "Advanced Pocket Computer", | ||||
|     "computercraft:pocket_computer_normal": "Pocket Computer", | ||||
|     "computercraft:printed_book": "Printed Book", | ||||
|     "computercraft:printed_page": "Printed Page", | ||||
|     "computercraft:printed_pages": "Printed Pages", | ||||
|     "computercraft:printer": "Printer", | ||||
|     "computercraft:speaker": "Speaker", | ||||
|     "computercraft:treasure_disk": "Floppy Disk", | ||||
|     "computercraft:turtle_advanced": "Advanced Turtle", | ||||
|     "computercraft:turtle_normal": "Turtle", | ||||
|     "computercraft:wired_modem": "Wired Modem", | ||||
|     "computercraft:wired_modem_full": "Wired Modem", | ||||
|     "computercraft:wireless_modem_advanced": "Ender Modem", | ||||
|     "computercraft:wireless_modem_normal": "Wireless Modem", | ||||
|     "minecraft:black_dye": "Black Dye", | ||||
|     "minecraft:blue_dye": "Blue Dye", | ||||
|     "minecraft:brown_dye": "Brown Dye", | ||||
|     "minecraft:chest": "Chest", | ||||
|     "minecraft:command_block": "Command Block", | ||||
|     "minecraft:cyan_dye": "Cyan Dye", | ||||
|     "minecraft:ender_eye": "Eye of Ender", | ||||
|     "minecraft:ender_pearl": "Ender Pearl", | ||||
|     "minecraft:glass_pane": "Glass Pane", | ||||
|     "minecraft:gold_block": "Block of Gold", | ||||
|     "minecraft:gold_ingot": "Gold Ingot", | ||||
|     "minecraft:golden_apple": "Golden Apple", | ||||
|     "minecraft:gray_dye": "Gray Dye", | ||||
|     "minecraft:green_dye": "Green Dye", | ||||
|     "minecraft:iron_ingot": "Iron Ingot", | ||||
|     "minecraft:leather": "Leather", | ||||
|     "minecraft:light_blue_dye": "Light Blue Dye", | ||||
|     "minecraft:light_gray_dye": "Light Gray Dye", | ||||
|     "minecraft:lime_dye": "Lime Dye", | ||||
|     "minecraft:magenta_dye": "Magenta Dye", | ||||
|     "minecraft:note_block": "Note Block", | ||||
|     "minecraft:orange_dye": "Orange Dye", | ||||
|     "minecraft:pink_dye": "Pink Dye", | ||||
|     "minecraft:purple_dye": "Purple Dye", | ||||
|     "minecraft:red_dye": "Red Dye", | ||||
|     "minecraft:redstone": "Redstone Dust", | ||||
|     "minecraft:stone": "Stone", | ||||
|     "minecraft:string": "String", | ||||
|     "minecraft:white_dye": "White Dye", | ||||
|     "minecraft:yellow_dye": "Yellow Dye" | ||||
|   }, | ||||
|   "recipes": { | ||||
|     "computercraft:cable": { | ||||
|       "inputs": [ | ||||
|         null, | ||||
|         [ | ||||
|           "minecraft:stone" | ||||
|         ], | ||||
|         null, | ||||
|         [ | ||||
|           "minecraft:stone" | ||||
|         ], | ||||
|         [ | ||||
|           "minecraft:redstone" | ||||
|         ], | ||||
|         [ | ||||
|           "minecraft:stone" | ||||
|         ], | ||||
|         null, | ||||
|         [ | ||||
|           "minecraft:stone" | ||||
|         ], | ||||
|         null | ||||
|       ], | ||||
|       "output": "computercraft:cable", | ||||
|       "count": 6 | ||||
|     }, | ||||
|     "computercraft:computer_advanced": { | ||||
|       "inputs": [ | ||||
|         [ | ||||
|           "minecraft:gold_ingot" | ||||
|         ], | ||||
|         [ | ||||
|           "minecraft:gold_ingot" | ||||
|         ], | ||||
|         [ | ||||
|           "minecraft:gold_ingot" | ||||
|         ], | ||||
|         [ | ||||
|           "minecraft:gold_ingot" | ||||
|         ], | ||||
|         [ | ||||
|           "minecraft:redstone" | ||||
|         ], | ||||
|         [ | ||||
|           "minecraft:gold_ingot" | ||||
|         ], | ||||
|         [ | ||||
|           "minecraft:gold_ingot" | ||||
|         ], | ||||
|         [ | ||||
|           "minecraft:glass_pane" | ||||
|         ], | ||||
|         [ | ||||
|           "minecraft:gold_ingot" | ||||
|         ] | ||||
|       ], | ||||
|       "output": "computercraft:computer_advanced", | ||||
|       "count": 1 | ||||
|     }, | ||||
|     "computercraft:computer_advanced_upgrade": { | ||||
|       "inputs": [ | ||||
|         [ | ||||
|           "minecraft:gold_ingot" | ||||
|         ], | ||||
|         [ | ||||
|           "minecraft:gold_ingot" | ||||
|         ], | ||||
|         [ | ||||
|           "minecraft:gold_ingot" | ||||
|         ], | ||||
|         [ | ||||
|           "minecraft:gold_ingot" | ||||
|         ], | ||||
|         [ | ||||
|           "computercraft:computer_normal" | ||||
|         ], | ||||
|         [ | ||||
|           "minecraft:gold_ingot" | ||||
|         ], | ||||
|         [ | ||||
|           "minecraft:gold_ingot" | ||||
|         ], | ||||
|         null, | ||||
|         [ | ||||
|           "minecraft:gold_ingot" | ||||
|         ] | ||||
|       ], | ||||
|       "output": "computercraft:computer_advanced", | ||||
|       "count": 1 | ||||
|     }, | ||||
|     "computercraft:computer_command": { | ||||
|       "inputs": [ | ||||
|         [ | ||||
|           "minecraft:gold_ingot" | ||||
|         ], | ||||
|         [ | ||||
|           "minecraft:gold_ingot" | ||||
|         ], | ||||
|         [ | ||||
|           "minecraft:gold_ingot" | ||||
|         ], | ||||
|         [ | ||||
|           "minecraft:gold_ingot" | ||||
|         ], | ||||
|         [ | ||||
|           "minecraft:command_block" | ||||
|         ], | ||||
|         [ | ||||
|           "minecraft:gold_ingot" | ||||
|         ], | ||||
|         [ | ||||
|           "minecraft:gold_ingot" | ||||
|         ], | ||||
|         [ | ||||
|           "minecraft:glass_pane" | ||||
|         ], | ||||
|         [ | ||||
|           "minecraft:gold_ingot" | ||||
|         ] | ||||
|       ], | ||||
|       "output": "computercraft:computer_command", | ||||
|       "count": 1 | ||||
|     }, | ||||
|     "computercraft:computer_normal": { | ||||
|       "inputs": [ | ||||
|         [ | ||||
|           "minecraft:stone" | ||||
|         ], | ||||
|         [ | ||||
|           "minecraft:stone" | ||||
|         ], | ||||
|         [ | ||||
|           "minecraft:stone" | ||||
|         ], | ||||
|         [ | ||||
|           "minecraft:stone" | ||||
|         ], | ||||
|         [ | ||||
|           "minecraft:redstone" | ||||
|         ], | ||||
|         [ | ||||
|           "minecraft:stone" | ||||
|         ], | ||||
|         [ | ||||
|           "minecraft:stone" | ||||
|         ], | ||||
|         [ | ||||
|           "minecraft:glass_pane" | ||||
|         ], | ||||
|         [ | ||||
|           "minecraft:stone" | ||||
|         ] | ||||
|       ], | ||||
|       "output": "computercraft:computer_normal", | ||||
|       "count": 1 | ||||
|     }, | ||||
|     "computercraft:disk_drive": { | ||||
|       "inputs": [ | ||||
|         [ | ||||
|           "minecraft:stone" | ||||
|         ], | ||||
|         [ | ||||
|           "minecraft:stone" | ||||
|         ], | ||||
|         [ | ||||
|           "minecraft:stone" | ||||
|         ], | ||||
|         [ | ||||
|           "minecraft:stone" | ||||
|         ], | ||||
|         [ | ||||
|           "minecraft:redstone" | ||||
|         ], | ||||
|         [ | ||||
|           "minecraft:stone" | ||||
|         ], | ||||
|         [ | ||||
|           "minecraft:stone" | ||||
|         ], | ||||
|         [ | ||||
|           "minecraft:redstone" | ||||
|         ], | ||||
|         [ | ||||
|           "minecraft:stone" | ||||
|         ] | ||||
|       ], | ||||
|       "output": "computercraft:disk_drive", | ||||
|       "count": 1 | ||||
|     }, | ||||
|     "computercraft:monitor_advanced": { | ||||
|       "inputs": [ | ||||
|         [ | ||||
|           "minecraft:gold_ingot" | ||||
|         ], | ||||
|         [ | ||||
|           "minecraft:gold_ingot" | ||||
|         ], | ||||
|         [ | ||||
|           "minecraft:gold_ingot" | ||||
|         ], | ||||
|         [ | ||||
|           "minecraft:gold_ingot" | ||||
|         ], | ||||
|         [ | ||||
|           "minecraft:glass_pane" | ||||
|         ], | ||||
|         [ | ||||
|           "minecraft:gold_ingot" | ||||
|         ], | ||||
|         [ | ||||
|           "minecraft:gold_ingot" | ||||
|         ], | ||||
|         [ | ||||
|           "minecraft:gold_ingot" | ||||
|         ], | ||||
|         [ | ||||
|           "minecraft:gold_ingot" | ||||
|         ] | ||||
|       ], | ||||
|       "output": "computercraft:monitor_advanced", | ||||
|       "count": 4 | ||||
|     }, | ||||
|     "computercraft:monitor_normal": { | ||||
|       "inputs": [ | ||||
|         [ | ||||
|           "minecraft:stone" | ||||
|         ], | ||||
|         [ | ||||
|           "minecraft:stone" | ||||
|         ], | ||||
|         [ | ||||
|           "minecraft:stone" | ||||
|         ], | ||||
|         [ | ||||
|           "minecraft:stone" | ||||
|         ], | ||||
|         [ | ||||
|           "minecraft:glass_pane" | ||||
|         ], | ||||
|         [ | ||||
|           "minecraft:stone" | ||||
|         ], | ||||
|         [ | ||||
|           "minecraft:stone" | ||||
|         ], | ||||
|         [ | ||||
|           "minecraft:stone" | ||||
|         ], | ||||
|         [ | ||||
|           "minecraft:stone" | ||||
|         ] | ||||
|       ], | ||||
|       "output": "computercraft:monitor_normal", | ||||
|       "count": 1 | ||||
|     }, | ||||
|     "computercraft:pocket_computer_advanced": { | ||||
|       "inputs": [ | ||||
|         [ | ||||
|           "minecraft:gold_ingot" | ||||
|         ], | ||||
|         [ | ||||
|           "minecraft:gold_ingot" | ||||
|         ], | ||||
|         [ | ||||
|           "minecraft:gold_ingot" | ||||
|         ], | ||||
|         [ | ||||
|           "minecraft:gold_ingot" | ||||
|         ], | ||||
|         [ | ||||
|           "minecraft:golden_apple" | ||||
|         ], | ||||
|         [ | ||||
|           "minecraft:gold_ingot" | ||||
|         ], | ||||
|         [ | ||||
|           "minecraft:gold_ingot" | ||||
|         ], | ||||
|         [ | ||||
|           "minecraft:glass_pane" | ||||
|         ], | ||||
|         [ | ||||
|           "minecraft:gold_ingot" | ||||
|         ] | ||||
|       ], | ||||
|       "output": "computercraft:pocket_computer_advanced", | ||||
|       "count": 1 | ||||
|     }, | ||||
|     "computercraft:pocket_computer_advanced_upgrade": { | ||||
|       "inputs": [ | ||||
|         [ | ||||
|           "minecraft:gold_ingot" | ||||
|         ], | ||||
|         [ | ||||
|           "minecraft:gold_ingot" | ||||
|         ], | ||||
|         [ | ||||
|           "minecraft:gold_ingot" | ||||
|         ], | ||||
|         [ | ||||
|           "minecraft:gold_ingot" | ||||
|         ], | ||||
|         [ | ||||
|           "computercraft:pocket_computer_normal" | ||||
|         ], | ||||
|         [ | ||||
|           "minecraft:gold_ingot" | ||||
|         ], | ||||
|         [ | ||||
|           "minecraft:gold_ingot" | ||||
|         ], | ||||
|         null, | ||||
|         [ | ||||
|           "minecraft:gold_ingot" | ||||
|         ] | ||||
|       ], | ||||
|       "output": "computercraft:pocket_computer_advanced", | ||||
|       "count": 1 | ||||
|     }, | ||||
|     "computercraft:pocket_computer_normal": { | ||||
|       "inputs": [ | ||||
|         [ | ||||
|           "minecraft:stone" | ||||
|         ], | ||||
|         [ | ||||
|           "minecraft:stone" | ||||
|         ], | ||||
|         [ | ||||
|           "minecraft:stone" | ||||
|         ], | ||||
|         [ | ||||
|           "minecraft:stone" | ||||
|         ], | ||||
|         [ | ||||
|           "minecraft:golden_apple" | ||||
|         ], | ||||
|         [ | ||||
|           "minecraft:stone" | ||||
|         ], | ||||
|         [ | ||||
|           "minecraft:stone" | ||||
|         ], | ||||
|         [ | ||||
|           "minecraft:glass_pane" | ||||
|         ], | ||||
|         [ | ||||
|           "minecraft:stone" | ||||
|         ] | ||||
|       ], | ||||
|       "output": "computercraft:pocket_computer_normal", | ||||
|       "count": 1 | ||||
|     }, | ||||
|     "computercraft:printed_book": { | ||||
|       "inputs": [ | ||||
|         [ | ||||
|           "minecraft:leather" | ||||
|         ], | ||||
|         [ | ||||
|           "computercraft:printed_page" | ||||
|         ], | ||||
|         [ | ||||
|           "minecraft:string" | ||||
|         ], | ||||
|         null, | ||||
|         null, | ||||
|         null, | ||||
|         null, | ||||
|         null, | ||||
|         null | ||||
|       ], | ||||
|       "output": "computercraft:printed_book", | ||||
|       "count": 1 | ||||
|     }, | ||||
|     "computercraft:printed_pages": { | ||||
|       "inputs": [ | ||||
|         [ | ||||
|           "computercraft:printed_page" | ||||
|         ], | ||||
|         [ | ||||
|           "computercraft:printed_page" | ||||
|         ], | ||||
|         [ | ||||
|           "minecraft:string" | ||||
|         ], | ||||
|         null, | ||||
|         null, | ||||
|         null, | ||||
|         null, | ||||
|         null, | ||||
|         null | ||||
|       ], | ||||
|       "output": "computercraft:printed_pages", | ||||
|       "count": 1 | ||||
|     }, | ||||
|     "computercraft:printer": { | ||||
|       "inputs": [ | ||||
|         [ | ||||
|           "minecraft:stone" | ||||
|         ], | ||||
|         [ | ||||
|           "minecraft:stone" | ||||
|         ], | ||||
|         [ | ||||
|           "minecraft:stone" | ||||
|         ], | ||||
|         [ | ||||
|           "minecraft:stone" | ||||
|         ], | ||||
|         [ | ||||
|           "minecraft:redstone" | ||||
|         ], | ||||
|         [ | ||||
|           "minecraft:stone" | ||||
|         ], | ||||
|         [ | ||||
|           "minecraft:stone" | ||||
|         ], | ||||
|         [ | ||||
|           "minecraft:black_dye", | ||||
|           "minecraft:blue_dye", | ||||
|           "minecraft:brown_dye", | ||||
|           "minecraft:cyan_dye", | ||||
|           "minecraft:gray_dye", | ||||
|           "minecraft:green_dye", | ||||
|           "minecraft:light_blue_dye", | ||||
|           "minecraft:light_gray_dye", | ||||
|           "minecraft:lime_dye", | ||||
|           "minecraft:magenta_dye", | ||||
|           "minecraft:orange_dye", | ||||
|           "minecraft:pink_dye", | ||||
|           "minecraft:purple_dye", | ||||
|           "minecraft:red_dye", | ||||
|           "minecraft:white_dye", | ||||
|           "minecraft:yellow_dye" | ||||
|         ], | ||||
|         [ | ||||
|           "minecraft:stone" | ||||
|         ] | ||||
|       ], | ||||
|       "output": "computercraft:printer", | ||||
|       "count": 1 | ||||
|     }, | ||||
|     "computercraft:speaker": { | ||||
|       "inputs": [ | ||||
|         [ | ||||
|           "minecraft:stone" | ||||
|         ], | ||||
|         [ | ||||
|           "minecraft:stone" | ||||
|         ], | ||||
|         [ | ||||
|           "minecraft:stone" | ||||
|         ], | ||||
|         [ | ||||
|           "minecraft:stone" | ||||
|         ], | ||||
|         [ | ||||
|           "minecraft:note_block" | ||||
|         ], | ||||
|         [ | ||||
|           "minecraft:stone" | ||||
|         ], | ||||
|         [ | ||||
|           "minecraft:stone" | ||||
|         ], | ||||
|         [ | ||||
|           "minecraft:redstone" | ||||
|         ], | ||||
|         [ | ||||
|           "minecraft:stone" | ||||
|         ] | ||||
|       ], | ||||
|       "output": "computercraft:speaker", | ||||
|       "count": 1 | ||||
|     }, | ||||
|     "computercraft:turtle_advanced": { | ||||
|       "inputs": [ | ||||
|         [ | ||||
|           "minecraft:gold_ingot" | ||||
|         ], | ||||
|         [ | ||||
|           "minecraft:gold_ingot" | ||||
|         ], | ||||
|         [ | ||||
|           "minecraft:gold_ingot" | ||||
|         ], | ||||
|         [ | ||||
|           "minecraft:gold_ingot" | ||||
|         ], | ||||
|         [ | ||||
|           "computercraft:computer_advanced" | ||||
|         ], | ||||
|         [ | ||||
|           "minecraft:gold_ingot" | ||||
|         ], | ||||
|         [ | ||||
|           "minecraft:gold_ingot" | ||||
|         ], | ||||
|         [ | ||||
|           "minecraft:chest" | ||||
|         ], | ||||
|         [ | ||||
|           "minecraft:gold_ingot" | ||||
|         ] | ||||
|       ], | ||||
|       "output": "computercraft:turtle_advanced", | ||||
|       "count": 1 | ||||
|     }, | ||||
|     "computercraft:turtle_advanced_upgrade": { | ||||
|       "inputs": [ | ||||
|         [ | ||||
|           "minecraft:gold_ingot" | ||||
|         ], | ||||
|         [ | ||||
|           "minecraft:gold_ingot" | ||||
|         ], | ||||
|         [ | ||||
|           "minecraft:gold_ingot" | ||||
|         ], | ||||
|         [ | ||||
|           "minecraft:gold_ingot" | ||||
|         ], | ||||
|         [ | ||||
|           "computercraft:turtle_normal" | ||||
|         ], | ||||
|         [ | ||||
|           "minecraft:gold_ingot" | ||||
|         ], | ||||
|         null, | ||||
|         [ | ||||
|           "minecraft:gold_block" | ||||
|         ], | ||||
|         null | ||||
|       ], | ||||
|       "output": "computercraft:turtle_advanced", | ||||
|       "count": 1 | ||||
|     }, | ||||
|     "computercraft:turtle_normal": { | ||||
|       "inputs": [ | ||||
|         [ | ||||
|           "minecraft:iron_ingot" | ||||
|         ], | ||||
|         [ | ||||
|           "minecraft:iron_ingot" | ||||
|         ], | ||||
|         [ | ||||
|           "minecraft:iron_ingot" | ||||
|         ], | ||||
|         [ | ||||
|           "minecraft:iron_ingot" | ||||
|         ], | ||||
|         [ | ||||
|           "computercraft:computer_normal" | ||||
|         ], | ||||
|         [ | ||||
|           "minecraft:iron_ingot" | ||||
|         ], | ||||
|         [ | ||||
|           "minecraft:iron_ingot" | ||||
|         ], | ||||
|         [ | ||||
|           "minecraft:chest" | ||||
|         ], | ||||
|         [ | ||||
|           "minecraft:iron_ingot" | ||||
|         ] | ||||
|       ], | ||||
|       "output": "computercraft:turtle_normal", | ||||
|       "count": 1 | ||||
|     }, | ||||
|     "computercraft:wired_modem": { | ||||
|       "inputs": [ | ||||
|         [ | ||||
|           "minecraft:stone" | ||||
|         ], | ||||
|         [ | ||||
|           "minecraft:stone" | ||||
|         ], | ||||
|         [ | ||||
|           "minecraft:stone" | ||||
|         ], | ||||
|         [ | ||||
|           "minecraft:stone" | ||||
|         ], | ||||
|         [ | ||||
|           "minecraft:redstone" | ||||
|         ], | ||||
|         [ | ||||
|           "minecraft:stone" | ||||
|         ], | ||||
|         [ | ||||
|           "minecraft:stone" | ||||
|         ], | ||||
|         [ | ||||
|           "minecraft:stone" | ||||
|         ], | ||||
|         [ | ||||
|           "minecraft:stone" | ||||
|         ] | ||||
|       ], | ||||
|       "output": "computercraft:wired_modem", | ||||
|       "count": 1 | ||||
|     }, | ||||
|     "computercraft:wired_modem_full_from": { | ||||
|       "inputs": [ | ||||
|         [ | ||||
|           "computercraft:wired_modem" | ||||
|         ], | ||||
|         null, | ||||
|         null, | ||||
|         null, | ||||
|         null, | ||||
|         null, | ||||
|         null, | ||||
|         null, | ||||
|         null | ||||
|       ], | ||||
|       "output": "computercraft:wired_modem_full", | ||||
|       "count": 1 | ||||
|     }, | ||||
|     "computercraft:wired_modem_full_to": { | ||||
|       "inputs": [ | ||||
|         [ | ||||
|           "computercraft:wired_modem_full" | ||||
|         ], | ||||
|         null, | ||||
|         null, | ||||
|         null, | ||||
|         null, | ||||
|         null, | ||||
|         null, | ||||
|         null, | ||||
|         null | ||||
|       ], | ||||
|       "output": "computercraft:wired_modem", | ||||
|       "count": 1 | ||||
|     }, | ||||
|     "computercraft:wireless_modem_advanced": { | ||||
|       "inputs": [ | ||||
|         [ | ||||
|           "minecraft:gold_ingot" | ||||
|         ], | ||||
|         [ | ||||
|           "minecraft:gold_ingot" | ||||
|         ], | ||||
|         [ | ||||
|           "minecraft:gold_ingot" | ||||
|         ], | ||||
|         [ | ||||
|           "minecraft:gold_ingot" | ||||
|         ], | ||||
|         [ | ||||
|           "minecraft:ender_eye" | ||||
|         ], | ||||
|         [ | ||||
|           "minecraft:gold_ingot" | ||||
|         ], | ||||
|         [ | ||||
|           "minecraft:gold_ingot" | ||||
|         ], | ||||
|         [ | ||||
|           "minecraft:gold_ingot" | ||||
|         ], | ||||
|         [ | ||||
|           "minecraft:gold_ingot" | ||||
|         ] | ||||
|       ], | ||||
|       "output": "computercraft:wireless_modem_advanced", | ||||
|       "count": 1 | ||||
|     }, | ||||
|     "computercraft:wireless_modem_normal": { | ||||
|       "inputs": [ | ||||
|         [ | ||||
|           "minecraft:stone" | ||||
|         ], | ||||
|         [ | ||||
|           "minecraft:stone" | ||||
|         ], | ||||
|         [ | ||||
|           "minecraft:stone" | ||||
|         ], | ||||
|         [ | ||||
|           "minecraft:stone" | ||||
|         ], | ||||
|         [ | ||||
|           "minecraft:ender_pearl" | ||||
|         ], | ||||
|         [ | ||||
|           "minecraft:stone" | ||||
|         ], | ||||
|         [ | ||||
|           "minecraft:stone" | ||||
|         ], | ||||
|         [ | ||||
|           "minecraft:stone" | ||||
|         ], | ||||
|         [ | ||||
|           "minecraft:stone" | ||||
|         ] | ||||
|       ], | ||||
|       "output": "computercraft:wireless_modem_normal", | ||||
|       "count": 1 | ||||
|     } | ||||
|   } | ||||
| } | ||||
| Before Width: | Height: | Size: 375 B | 
| Before Width: | Height: | Size: 2.5 KiB | 
| Before Width: | Height: | Size: 3.1 KiB | 
| Before Width: | Height: | Size: 1.2 KiB | 
| Before Width: | Height: | Size: 189 B | 
| Before Width: | Height: | Size: 1.3 KiB | 
| Before Width: | Height: | Size: 2.4 KiB | 
| Before Width: | Height: | Size: 1.1 KiB | 
| Before Width: | Height: | Size: 158 B |